From 4e202aa1a51a6621e815a6a6fd79d55a753764e6 Mon Sep 17 00:00:00 2001 From: Julien Danjou Date: Mon, 13 May 2013 18:09:05 +0200 Subject: [PATCH 1/6] Check that compiler error are user always user friendly Signed-off-by: Julien Danjou --- hy/compiler.py | 45 +++++++++++++++++++++++-------------- tests/compilers/test_ast.py | 9 +++++--- 2 files changed, 34 insertions(+), 20 deletions(-) diff --git a/hy/compiler.py b/hy/compiler.py index ea3c5de..60c5335 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -419,22 +419,24 @@ class HyASTCompiler(object): if isinstance(expr, HyLambdaListKeyword): if expr not in expr._valid_types: - raise HyCompileError("{0} is not a valid " - "lambda-keyword.".format(repr(expr))) + raise HyTypeError(expr, "{0} is not a valid " + "lambda-keyword.".format(repr(expr))) if expr == "&rest" and lambda_keyword is None: lambda_keyword = expr elif expr == "&optional": if len(defaults) > 0: - raise HyCompileError("There can only be &optional " - "arguments or one &key argument") + raise HyTypeError(expr, + "There can only be &optional " + "arguments or one &key argument") lambda_keyword = expr elif expr == "&key": lambda_keyword = expr elif expr == "&kwargs": lambda_keyword = expr else: - raise HyCompileError("{0} is in an invalid " - "position.".format(repr(expr))) + raise HyTypeError(expr, + "{0} is in an invalid " + "position.".format(repr(expr))) # we don't actually care about this token, so we set # our state and continue to the next token... continue @@ -443,17 +445,20 @@ class HyASTCompiler(object): args.append(expr) elif lambda_keyword == "&rest": if varargs: - raise HyCompileError("There can only be one " - "&rest argument") + raise HyTypeError(expr, + "There can only be one " + "&rest argument") varargs = str(expr) elif lambda_keyword == "&key": if type(expr) != HyDict: - raise TypeError("There can only be one &key " - "argument") + raise HyTypeError(expr, + "There can only be one &key " + "argument") else: if len(defaults) > 0: - raise HyCompileError("There can only be &optional " - "arguments or one &key argument") + raise HyTypeError(expr, + "There can only be &optional " + "arguments or one &key argument") # As you can see, Python has a funny way of # defining keyword arguments. for k, v in expr.items(): @@ -463,8 +468,9 @@ class HyASTCompiler(object): elif lambda_keyword == "&optional": if isinstance(expr, HyList): if not len(expr) == 2: - raise TypeError("optional args should be bare names " - "or 2-item lists") + raise HyTypeError(expr, + "optional args should be bare names " + "or 2-item lists") k, v = expr else: k = expr @@ -474,8 +480,9 @@ class HyASTCompiler(object): defaults.append(ret.force_expr) elif lambda_keyword == "&kwargs": if kwargs: - raise HyCompileError("There can only be one " - "&kwargs argument") + raise HyTypeError(expr, + "There can only be one " + "&kwargs argument") kwargs = str(expr) return ret, args, defaults, varargs, kwargs @@ -760,7 +767,8 @@ class HyASTCompiler(object): @builds("except") @builds("catch") def magic_internal_form(self, expr): - raise TypeError("Error: `%s' can't be used like that." % (expr[0])) + raise HyTypeError(expr, + "Error: `%s' can't be used like that." % (expr[0])) def _compile_catch_expression(self, expr, var): catch = expr.pop(0) # catch @@ -792,6 +800,9 @@ class HyASTCompiler(object): # let's pop variable and use it as name if len(exceptions) == 2: name = exceptions.pop(0) + if not isinstance(name, HySymbol): + raise HyTypeError(exceptions, + "Exception storage target name must be a symbol.") if sys.version_info[0] >= 3: # Python3 features a change where the Exception handler # moved the name from a Name() to a pure Python String type. diff --git a/tests/compilers/test_ast.py b/tests/compilers/test_ast.py index fc17892..694e10d 100644 --- a/tests/compilers/test_ast.py +++ b/tests/compilers/test_ast.py @@ -22,7 +22,7 @@ from __future__ import unicode_literals from hy import HyString -from hy.compiler import hy_compile, HyCompileError +from hy.compiler import hy_compile, HyCompileError, HyTypeError from hy.lex import tokenize import ast @@ -43,8 +43,11 @@ def cant_compile(expr): try: hy_compile(expr) assert False - except HyCompileError: - pass + except HyCompileError as e: + # Anything that can't be compiled should raise a user friendly + # error, otherwise it's a compiler bug. + assert isinstance(e.exception, HyTypeError) + assert e.traceback def test_ast_bad_type(): From 66e5af4a82099bb6d23a1e5c1f3cbc22459aeb5d Mon Sep 17 00:00:00 2001 From: Julien Danjou Date: Mon, 13 May 2013 18:44:50 +0200 Subject: [PATCH 2/6] Write a test for builds() check Signed-off-by: Julien Danjou --- hy/compiler.py | 2 +- tests/compilers/test_compiler.py | 16 +++++++++++++--- 2 files changed, 14 insertions(+), 4 deletions(-) diff --git a/hy/compiler.py b/hy/compiler.py index ea3c5de..7122a97 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -97,7 +97,7 @@ def builds(_type): unpythonic_chars = ["-"] really_ok = ["-"] - if True in (x in str_type(_type) for x in unpythonic_chars): + if any(x in unpythonic_chars for x in str_type(_type)): if _type not in really_ok: raise TypeError("`build' needs to be *post* translated strings, " "Mr. / Mrs. Hypser. -- `%s' sucks." % (_type)) diff --git a/tests/compilers/test_compiler.py b/tests/compilers/test_compiler.py index 5f93144..246534d 100644 --- a/tests/compilers/test_compiler.py +++ b/tests/compilers/test_compiler.py @@ -27,11 +27,21 @@ if sys.version_info[0] <= 2 and sys.version_info[1] <= 6: else: import unittest - +from hy import compiler from hy.models.expression import HyExpression from hy.models.list import HyList from hy.models.symbol import HySymbol -from hy.compiler import HyASTCompiler + + +class CompilerTest(unittest.TestCase): + + def test_builds_with_dash(self): + self.assert_(callable(compiler.builds("foobar"))) + self.assert_(callable(compiler.builds("foo_bar"))) + self.assert_(callable(compiler.builds("-"))) + self.assertRaisesRegexp(TypeError, + "\*post\* translated strings", + compiler.builds, "foobar-with-dash-") class HyASTCompilerTest(unittest.TestCase): @@ -46,7 +56,7 @@ class HyASTCompilerTest(unittest.TestCase): return h def setUp(self): - self.c = HyASTCompiler() + self.c = compiler.HyASTCompiler() def test_fn_compiler_empty_function(self): ret = self.c.compile_function_def( From 9e03e0e6ec796ccb6713a0b8f7dc3c3b014bafc5 Mon Sep 17 00:00:00 2001 From: Nicolas Dandrimont Date: Tue, 14 May 2013 11:39:20 +0200 Subject: [PATCH 3/6] Modify HyDict semantics to allow nesting expressions HyDicts are now HyLists, that get compiled down to dicts only by the compiler. --- hy/compiler.py | 7 ++++--- hy/lex/states.py | 4 +--- hy/macros.py | 8 +------- hy/models/dict.py | 14 +++++--------- tests/lex/test_lex.py | 9 +++------ 5 files changed, 14 insertions(+), 28 deletions(-) diff --git a/hy/compiler.py b/hy/compiler.py index ea3c5de..00931db 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -456,7 +456,8 @@ class HyASTCompiler(object): "arguments or one &key argument") # As you can see, Python has a funny way of # defining keyword arguments. - for k, v in expr.items(): + it = iter(expr) + for k, v in zip(it, it): args.append(k) ret += self.compile(v) defaults.append(ret.force_expr) @@ -543,7 +544,7 @@ class HyASTCompiler(object): name = form.__class__.__name__ imports = set([name]) - if isinstance(form, HyList): + if isinstance(form, (HyList, HyDict)): if not form: contents = HyList() else: @@ -1685,7 +1686,7 @@ class HyASTCompiler(object): @builds(HyDict) def compile_dict(self, m): - keyvalues, ret = self._compile_collect(sum(m.items(), ())) + keyvalues, ret = self._compile_collect(m) ret += ast.Dict(lineno=m.start_line, col_offset=m.start_column, diff --git a/hy/lex/states.py b/hy/lex/states.py index 772f8c6..51255bd 100644 --- a/hy/lex/states.py +++ b/hy/lex/states.py @@ -228,9 +228,7 @@ class Dict(ListeyThing): def exit(self): self.commit() - it = iter(self.nodes) - result = dict(zip(it, it)) - self.result = HyDict(result) + self.result = HyDict(self.nodes) end_char = "}" diff --git a/hy/macros.py b/hy/macros.py index b58d0ec..615e2dc 100644 --- a/hy/macros.py +++ b/hy/macros.py @@ -20,7 +20,6 @@ from hy.models.expression import HyExpression from hy.models.string import HyString -from hy.models.dict import HyDict from hy.models.list import HyList _hy_macros = {} @@ -51,13 +50,8 @@ def process(tree): ntree.replace(tree) return ntree - if isinstance(tree, HyDict): - obj = HyDict(dict((process(x), process(tree[x])) for x in tree)) - obj.replace(tree) - return obj - if isinstance(tree, HyList): - obj = HyList([process(x) for x in tree]) # NOQA + obj = tree.__class__([process(x) for x in tree]) # NOQA # flake8 thinks we're redefining from 52. obj.replace(tree) return obj diff --git a/hy/models/dict.py b/hy/models/dict.py index 43384bd..1bf020c 100644 --- a/hy/models/dict.py +++ b/hy/models/dict.py @@ -18,17 +18,13 @@ # FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER # DEALINGS IN THE SOFTWARE. -from hy.models import HyObject +from hy.models.list import HyList -class HyDict(HyObject, dict): +class HyDict(HyList): """ - HyDict (just a dict) + HyDict (just a representation of a dict) """ - def replace(self, other): - for x in self: - self[x].replace(other) - x.replace(other) - - HyObject.replace(self, other) + def __repr__(self): + return "{%s}" % (" ".join([repr(x) for x in self])) diff --git a/tests/lex/test_lex.py b/tests/lex/test_lex.py index 590e51e..490e167 100644 --- a/tests/lex/test_lex.py +++ b/tests/lex/test_lex.py @@ -185,15 +185,12 @@ def test_lex_line_counting_multi_inner(): def test_dicts(): """ Ensure that we can tokenize a dict. """ objs = tokenize("{foo bar bar baz}") - assert objs == [HyDict({ - "foo": "bar", - "bar": "baz" - })] + assert objs == [HyDict(["foo", "bar", "bar", "baz"])] objs = tokenize("(bar {foo bar bar baz})") assert objs == [HyExpression([HySymbol("bar"), - HyDict({"foo": "bar", - "bar": "baz"})])] + HyDict(["foo", "bar", + "bar", "baz"])])] def test_nospace(): From 3252af312911fd617d205ec091adbe115193315c Mon Sep 17 00:00:00 2001 From: Nicolas Dandrimont Date: Tue, 14 May 2013 12:01:23 +0200 Subject: [PATCH 4/6] Add tests for expr-in-dict --- tests/lex/test_lex.py | 6 ++++++ tests/native_tests/language.hy | 6 ++++++ 2 files changed, 12 insertions(+) diff --git a/tests/lex/test_lex.py b/tests/lex/test_lex.py index 490e167..7b90750 100644 --- a/tests/lex/test_lex.py +++ b/tests/lex/test_lex.py @@ -192,6 +192,12 @@ def test_dicts(): HyDict(["foo", "bar", "bar", "baz"])])] + objs = tokenize("{(foo bar) (baz quux)}") + assert objs == [HyDict([ + HyExpression([HySymbol("foo"), HySymbol("bar")]), + HyExpression([HySymbol("baz"), HySymbol("quux")]) + ])] + def test_nospace(): """ Ensure we can tokenize without spaces if we have to """ diff --git a/tests/native_tests/language.hy b/tests/native_tests/language.hy index 0254493..3658012 100644 --- a/tests/native_tests/language.hy +++ b/tests/native_tests/language.hy @@ -15,6 +15,12 @@ (assert (= [1 2 3 4] (+ [1 2] [3 4])))) +(defn test-dicts [] + "NATIVE: test dicts work right" + (assert (= {1 2 3 4} {3 4 1 2})) + (assert (= {1 2 3 4} {1 (+ 1 1) 3 (+ 2 2)}))) + + (defn test-setv-get [] "NATIVE: test setv works on a get expression" (setv foo [0 1 2]) From f939ae95444fd3d4caef2e23e4fd54c56a5ab3ab Mon Sep 17 00:00:00 2001 From: Nicolas Dandrimont Date: Tue, 14 May 2013 12:10:29 +0200 Subject: [PATCH 5/6] Test quoting dicts --- tests/native_tests/quote.hy | 22 ++++++++++++++++++++++ 1 file changed, 22 insertions(+) diff --git a/tests/native_tests/quote.hy b/tests/native_tests/quote.hy index 0b64f3d..f180a98 100644 --- a/tests/native_tests/quote.hy +++ b/tests/native_tests/quote.hy @@ -1,3 +1,6 @@ +(import hy) + + (defn test-quote [] "NATIVE: test for quoting functionality" (setf q (quote (a b c))) @@ -21,6 +24,25 @@ (assert (= (cdr q1) (quote (a b c))))) +(defn test-quote-dicts [] + "NATIVE: test quoting dicts" + (setf q (quote {foo bar baz quux})) + (assert (= (len q) 4)) + (assert (= (get q 0) (quote foo))) + (assert (= (get q 1) (quote bar))) + (assert (= (get q 2) (quote baz))) + (assert (= (get q 3) (quote quux))) + (assert (= (type q) hy.HyDict))) + + +(defn test-quote-expr-in-dict [] + "NATIVE: test quoting nested exprs in dict" + (setf q (quote {(foo bar) 0})) + (assert (= (len q) 2)) + (setf qq (get q 0)) + (assert (= qq (quote (foo bar))))) + + (defn test-quasiquote [] "NATIVE: test that quasiquote and quote are equivalent for simple cases" (setf q (quote (a b c))) From 81096596c71a0ad50124a1519cddc385d14a7df4 Mon Sep 17 00:00:00 2001 From: Paul Tagliamonte Date: Fri, 17 May 2013 11:07:36 -0400 Subject: [PATCH 6/6] style fix - long line --- hy/compiler.py | 6 ++++-- 1 file changed, 4 insertions(+), 2 deletions(-) diff --git a/hy/compiler.py b/hy/compiler.py index 60c5335..6a99ece 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -801,8 +801,10 @@ class HyASTCompiler(object): if len(exceptions) == 2: name = exceptions.pop(0) if not isinstance(name, HySymbol): - raise HyTypeError(exceptions, - "Exception storage target name must be a symbol.") + raise HyTypeError( + exceptions, + "Exception storage target name must be a symbol.") + if sys.version_info[0] >= 3: # Python3 features a change where the Exception handler # moved the name from a Name() to a pure Python String type.