From ceedc69b7d505d76bb4a20541891142b4b6e0a8e Mon Sep 17 00:00:00 2001 From: Paul Tagliamonte Date: Wed, 8 May 2013 19:58:36 -0400 Subject: [PATCH 01/16] Correct existing tests for future try / except work. --- tests/compilers/test_ast.py | 42 ++++++++++++++++++------------------- 1 file changed, 21 insertions(+), 21 deletions(-) diff --git a/tests/compilers/test_ast.py b/tests/compilers/test_ast.py index 958c7b8..cd85c59 100644 --- a/tests/compilers/test_ast.py +++ b/tests/compilers/test_ast.py @@ -140,39 +140,39 @@ def test_ast_bad_try(): def test_ast_good_catch(): "Make sure AST can compile valid catch" - hy_compile(tokenize("(catch)")) - hy_compile(tokenize("(catch [])")) - hy_compile(tokenize("(catch [Foobar])")) - hy_compile(tokenize("(catch [[]])")) - hy_compile(tokenize("(catch [x FooBar])")) - hy_compile(tokenize("(catch [x [FooBar BarFoo]])")) - hy_compile(tokenize("(catch [x [FooBar BarFoo]])")) + hy_compile(tokenize("(try 1 (catch))")) + hy_compile(tokenize("(try 1 (catch []))")) + hy_compile(tokenize("(try 1 (catch [Foobar]))")) + hy_compile(tokenize("(try 1 (catch [[]]))")) + hy_compile(tokenize("(try 1 (catch [x FooBar]))")) + hy_compile(tokenize("(try 1 (catch [x [FooBar BarFoo]]))")) + hy_compile(tokenize("(try 1 (catch [x [FooBar BarFoo]]))")) def test_ast_bad_catch(): "Make sure AST can't compile invalid catch" - cant_compile("(catch 1)") - cant_compile("(catch \"A\")") - cant_compile("(catch [1 3])") - cant_compile("(catch [x [FooBar] BarBar])") + cant_compile("(try (catch 1))") + cant_compile("(try (catch \"A\"))") + cant_compile("(try (catch [1 3]))") + cant_compile("(try (catch [x [FooBar] BarBar]))") def test_ast_good_except(): "Make sure AST can compile valid except" - hy_compile(tokenize("(except)")) - hy_compile(tokenize("(except [])")) - hy_compile(tokenize("(except [Foobar])")) - hy_compile(tokenize("(except [[]])")) - hy_compile(tokenize("(except [x FooBar])")) - hy_compile(tokenize("(except [x [FooBar BarFoo]])")) - hy_compile(tokenize("(except [x [FooBar BarFoo]])")) + hy_compile(tokenize("(try 1 (except))")) + hy_compile(tokenize("(try 1 (except []))")) + hy_compile(tokenize("(try 1 (except [Foobar]))")) + hy_compile(tokenize("(try 1 (except [[]]))")) + hy_compile(tokenize("(try 1 (except [x FooBar]))")) + hy_compile(tokenize("(try 1 (except [x [FooBar BarFoo]]))")) + hy_compile(tokenize("(try 1 (except [x [FooBar BarFoo]]))")) def test_ast_bad_except(): "Make sure AST can't compile invalid except" - cant_compile("(except 1)") - cant_compile("(except [1 3])") - cant_compile("(except [x [FooBar] BarBar])") + cant_compile("(try 1 (except 1))") + cant_compile("(try 1 (except [1 3]))") + cant_compile("(try 1 (except [x [FooBar] BarBar]))") def test_ast_good_assert(): From 058197a24f46bba677883fd56c4a7c0b793a4a49 Mon Sep 17 00:00:00 2001 From: Paul Tagliamonte Date: Wed, 8 May 2013 20:00:09 -0400 Subject: [PATCH 02/16] Allow returning inside a try / except. Closes #163 --- hy/compiler.py | 49 +++++++++++++++++++++++++++------- tests/compilers/test_ast.py | 2 ++ tests/native_tests/language.hy | 4 +++ 3 files changed, 45 insertions(+), 10 deletions(-) diff --git a/hy/compiler.py b/hy/compiler.py index dd6af76..1b5f810 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -557,8 +557,24 @@ class HyASTCompiler(object): # (try something…) body = self.compile(body) - # XXX we will likely want to make this a tempvar - body += body.expr_as_stmt() + var = self.get_anon_var() + name = ast.Name(id=ast_str(var), arg=ast_str(var), + ctx=ast.Store(), + lineno=expr.start_line, + col_offset=expr.start_column) + + expr_name = ast.Name(id=ast_str(var), arg=ast_str(var), + ctx=ast.Load(), + lineno=expr.start_line, + col_offset=expr.start_column) + + returnable = Result(expr=expr_name, temp_variables=[expr_name, name]) + + body += ast.Assign(targets=[name], + value=body.force_expr, + lineno=expr.start_line, + col_offset=expr.start_column) + body = body.stmts if not body: body = [ast.Pass(lineno=expr.start_line, @@ -574,7 +590,7 @@ class HyASTCompiler(object): raise HyTypeError(e, "Empty list not allowed in `try'") if e[0] in (HySymbol("except"), HySymbol("catch")): - handler_results += self.compile(e) + handler_results += self._compile_catch_expression(e, var) handlers.append(handler_results.stmts.pop()) elif e[0] == HySymbol("else"): if orelse: @@ -626,7 +642,7 @@ class HyASTCompiler(object): body=body, handlers=handlers, orelse=orelse, - finalbody=finalbody) + finalbody=finalbody) + returnable if finalbody: if handlers: @@ -639,30 +655,39 @@ class HyASTCompiler(object): handlers=handlers, body=body, orelse=orelse)], - finalbody=finalbody) + finalbody=finalbody) + returnable return ret + ast.TryFinally( lineno=expr.start_line, col_offset=expr.start_column, body=body, - finalbody=finalbody) + finalbody=finalbody) + returnable return ret + ast.TryExcept( lineno=expr.start_line, col_offset=expr.start_column, handlers=handlers, body=body, - orelse=orelse) + orelse=orelse) + returnable - @builds("catch") @builds("except") - def compile_catch_expression(self, expr): + @builds("catch") + def magic_internal_form(self, expr): + raise TypeError("Error: `%s' can't be used like that." % (expr[0])) + + def _compile_catch_expression(self, expr, var): catch = expr.pop(0) # catch + ret_name = ast.Name(id=ast_str(var), arg=ast_str(var), + ctx=ast.Store(), + lineno=expr.start_line, + col_offset=expr.start_column) + try: exceptions = expr.pop(0) except IndexError: exceptions = HyList() + # exceptions catch should be either: # [[list of exceptions]] # or @@ -673,6 +698,7 @@ class HyASTCompiler(object): # [exception] # or # [] + if not isinstance(exceptions, HyList): raise HyTypeError(exceptions, "`%s' exceptions list is not a list" % catch) @@ -720,7 +746,10 @@ class HyASTCompiler(object): "`%s' needs a valid exception list" % catch) body = self._compile_branch(expr) - # XXX tempvar handling magic + body += ast.Assign(targets=[ret_name], + value=body.force_expr, + lineno=expr.start_line, + col_offset=expr.start_column) body += body.expr_as_stmt() body = body.stmts diff --git a/tests/compilers/test_ast.py b/tests/compilers/test_ast.py index cd85c59..8fd7cb6 100644 --- a/tests/compilers/test_ast.py +++ b/tests/compilers/test_ast.py @@ -151,6 +151,7 @@ def test_ast_good_catch(): def test_ast_bad_catch(): "Make sure AST can't compile invalid catch" + cant_compile("(catch 22)") # heh cant_compile("(try (catch 1))") cant_compile("(try (catch \"A\"))") cant_compile("(try (catch [1 3]))") @@ -170,6 +171,7 @@ def test_ast_good_except(): def test_ast_bad_except(): "Make sure AST can't compile invalid except" + cant_compile("(except 1)") cant_compile("(try 1 (except 1))") cant_compile("(try 1 (except [1 3]))") cant_compile("(try 1 (except [x [FooBar] BarBar]))") diff --git a/tests/native_tests/language.hy b/tests/native_tests/language.hy index d4ec580..0a30d57 100644 --- a/tests/native_tests/language.hy +++ b/tests/native_tests/language.hy @@ -665,3 +665,7 @@ (if (if 0 True False) 42 43)))) + +(defn test-try-except-return [] + "NATIVE: test we can return from in a try except" + (assert ((fn [] (try xxx (except [NameError] (+ 1 1)))))) 2) From 1fa53f9255491fd17b2e589784bde6ef86827bfe Mon Sep 17 00:00:00 2001 From: Paul Tagliamonte Date: Wed, 8 May 2013 20:41:16 -0400 Subject: [PATCH 03/16] Try/Except: Ensure that we return properly From both inside a try and in an exception handler. --- hy/compiler.py | 9 ++------- tests/native_tests/language.hy | 6 +++++- 2 files changed, 7 insertions(+), 8 deletions(-) diff --git a/hy/compiler.py b/hy/compiler.py index 7e384a2..286925d 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -606,7 +606,7 @@ class HyASTCompiler(object): raise HyTypeError(e, "Empty list not allowed in `try'") if e[0] in (HySymbol("except"), HySymbol("catch")): - handler_results += self._compile_catch_expression(e, var) + handler_results += self._compile_catch_expression(e, name) handlers.append(handler_results.stmts.pop()) elif e[0] == HySymbol("else"): if orelse: @@ -694,11 +694,6 @@ class HyASTCompiler(object): def _compile_catch_expression(self, expr, var): catch = expr.pop(0) # catch - ret_name = ast.Name(id=ast_str(var), arg=ast_str(var), - ctx=ast.Store(), - lineno=expr.start_line, - col_offset=expr.start_column) - try: exceptions = expr.pop(0) except IndexError: @@ -762,7 +757,7 @@ class HyASTCompiler(object): "`%s' needs a valid exception list" % catch) body = self._compile_branch(expr) - body += ast.Assign(targets=[ret_name], + body += ast.Assign(targets=[var], value=body.force_expr, lineno=expr.start_line, col_offset=expr.start_column) diff --git a/tests/native_tests/language.hy b/tests/native_tests/language.hy index 59bba3a..c12a738 100644 --- a/tests/native_tests/language.hy +++ b/tests/native_tests/language.hy @@ -684,4 +684,8 @@ (defn test-try-except-return [] "NATIVE: test we can return from in a try except" - (assert ((fn [] (try xxx (except [NameError] (+ 1 1)))))) 2) + (assert (= ((fn [] (try xxx (except [NameError] (+ 1 1))))) 2)) + (setf foo (try xxx (except [NameError] (+ 1 1)))) + (assert (= foo 2)) + (setf foo (try (+ 2 2) (except [NameError] (+ 1 1)))) + (assert (= foo 4))) From d76d011b115d4d496fda5aa552c1601d61ae1172 Mon Sep 17 00:00:00 2001 From: "Clinton N. Dreisbach" Date: Wed, 8 May 2013 22:16:03 -0400 Subject: [PATCH 04/16] Added documentation about import --- docs/language/api.rst | 30 ++++++++++++++++++++++++++++++ 1 file changed, 30 insertions(+) diff --git a/docs/language/api.rst b/docs/language/api.rst index a59a450..41ea609 100644 --- a/docs/language/api.rst +++ b/docs/language/api.rst @@ -38,6 +38,36 @@ Hy features a number special forms that are used to help generate correct Python AST. The following are "special" forms, which may have behavior that's slightly unexpected in some situations. +import +------ + +`import` is used to import modules, like in Python. There are several forms +of import you can use. + +.. code-block:: clj + ;; Imports each of these modules + ;; + ;; Python: + ;; import sys + ;; import os.path + (import sys os.path) + + ;; Import from a module + ;; + ;; Python: from os.path import exists, isdir, isfile + (import [os.path [exists isdir isfile]]) + + ;; Import with an alias + ;; + ;; Python: import sys as systest + (import [sys :as systest]) + + ;; You can list as many imports as you like of different types. + (import [tests.resources [kwtest function-with-a-dash]] + [os.path [exists isdir isfile]] + [sys :as systest]) + + do / progn ---------- From 14d33633e10bba97cad504390282a510fcef794a Mon Sep 17 00:00:00 2001 From: Paul Tagliamonte Date: Wed, 8 May 2013 22:32:11 -0400 Subject: [PATCH 05/16] RST syntax fix --- docs/language/api.rst | 1 + 1 file changed, 1 insertion(+) diff --git a/docs/language/api.rst b/docs/language/api.rst index 41ea609..8f63838 100644 --- a/docs/language/api.rst +++ b/docs/language/api.rst @@ -45,6 +45,7 @@ import of import you can use. .. code-block:: clj + ;; Imports each of these modules ;; ;; Python: From 60c1a1ba068f4d44396be6687453fec1750103d6 Mon Sep 17 00:00:00 2001 From: Christopher Allan Webber Date: Thu, 9 May 2013 15:11:42 -0500 Subject: [PATCH 06/16] Docs: "Much more readable, no! -> Much more readable, no?" --- docs/tutorial.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/tutorial.rst b/docs/tutorial.rst index 1db2144..019564b 100644 --- a/docs/tutorial.rst +++ b/docs/tutorial.rst @@ -414,7 +414,7 @@ Which, of course, expands out to: (wc (grep (cat "/usr/share/dict/words") "-E" "^hy") "-l") -Much more readable, no! Use the threading macro! +Much more readable, no? Use the threading macro! TODO From b6d730c0447c116085cdad5de7e51067167c451d Mon Sep 17 00:00:00 2001 From: Christopher Allan Webber Date: Thu, 9 May 2013 15:35:47 -0500 Subject: [PATCH 07/16] Documenting: tuples and argument formatting --- docs/tutorial.rst | 48 +++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 48 insertions(+) diff --git a/docs/tutorial.rst b/docs/tutorial.rst index 019564b..3e83834 100644 --- a/docs/tutorial.rst +++ b/docs/tutorial.rst @@ -188,6 +188,8 @@ hy. Let's experiment with this in the hy interpreter:: ... "cat" "meow"} ... {'dog': 'bark', 'cat': 'meow'} + => (, 1 2 3) + (1, 2, 3) (You may notice that at present, the common lisp method of quoting things like so: @@ -375,6 +377,52 @@ In hy, you could do these like: ; (8, 'A'), (8, 'B'), (8, 'C'), (8, 'D'), (8, 'E'), (8, 'F'), (8, 'G'), (8, 'H')] +Python has support for various fancy argument and keyword arguments. +In python we might see:: + + >>> def optional_arg(pos1, pos2, keyword1=None, keyword2=42): + ... return [pos1, pos2, keyword1, keyword2] + ... + >>> optional_arg(1, 2) + [1, 2, None, 42] + >>> optional_arg(1, 2, 3, 4) + [1, 2, 3, 4] + >>> optional_arg(keyword1=1, pos2=2, pos1=3, keyword2=4) + [3, 2, 1, 4] + +The same thing in Hy: + +.. code-block:: clj + + => (defn optional_arg [pos1 pos2 &optional keyword1 [keyword2 88]] + ... [pos1 pos2 keyword1 keyword2]) + => (optional_arg 1 2) + [1 2 None 42] + => (optional_arg 1 2 3 4) + [1 2 3 4] + => (kwapply (optional_arg) + ... {"keyword1" 1 + ... "pos2" 2 + ... "pos1" 3 + ... "keyword2" 4}) + ... + [3, 2, 1, 4] + +See how we use kwapply to handle the fancy pssing? :) + +Hy also supports **args and **kwargs. In Python:: + + def some_func(foo, bar, *args, **kwargs): + import pprint + pprint.pprint((foo, bar, args, kwargs)) + +The Hy equivalent: + + (defn some_func [foo bar &rest args &kwargs kwargs] + (import pprint) + (pprint.pprint (, foo bar args kwargs))) + + Protips! ======== From 0a362a2120ae5e9891cddb711bed040b373477e0 Mon Sep 17 00:00:00 2001 From: Christopher Allan Webber Date: Thu, 9 May 2013 15:40:32 -0500 Subject: [PATCH 08/16] oh yeah, make this a clojure syntax block in the docs --- docs/tutorial.rst | 2 ++ 1 file changed, 2 insertions(+) diff --git a/docs/tutorial.rst b/docs/tutorial.rst index 3e83834..24e8ec6 100644 --- a/docs/tutorial.rst +++ b/docs/tutorial.rst @@ -418,6 +418,8 @@ Hy also supports **args and **kwargs. In Python:: The Hy equivalent: +.. code-block:: clj + (defn some_func [foo bar &rest args &kwargs kwargs] (import pprint) (pprint.pprint (, foo bar args kwargs))) From 5bd5620dd028853bb544a551112e545d040906f2 Mon Sep 17 00:00:00 2001 From: Christopher Allan Webber Date: Thu, 9 May 2013 15:48:02 -0500 Subject: [PATCH 09/16] Dictionary style keyword arguments documentation --- docs/tutorial.rst | 15 ++++++++++++--- 1 file changed, 12 insertions(+), 3 deletions(-) diff --git a/docs/tutorial.rst b/docs/tutorial.rst index 24e8ec6..99f18fc 100644 --- a/docs/tutorial.rst +++ b/docs/tutorial.rst @@ -390,9 +390,7 @@ In python we might see:: >>> optional_arg(keyword1=1, pos2=2, pos1=3, keyword2=4) [3, 2, 1, 4] -The same thing in Hy: - -.. code-block:: clj +The same thing in Hy:: => (defn optional_arg [pos1 pos2 &optional keyword1 [keyword2 88]] ... [pos1 pos2 keyword1 keyword2]) @@ -410,6 +408,17 @@ The same thing in Hy: See how we use kwapply to handle the fancy pssing? :) +There's also a dictionary-style keyword arguments construction that +looks like:: + +.. code-block:: clj + + (defn another_style [&key {"key1" "val1" "key2" "val2"}] + [key1 key2]) + +The difference here is that since it's a dictionary, you can't rely on +any specific ordering to the arguments. + Hy also supports **args and **kwargs. In Python:: def some_func(foo, bar, *args, **kwargs): From b4b3ab89f7f8042b7ad0c1c03c58fa0422e3f596 Mon Sep 17 00:00:00 2001 From: Christopher Allan Webber Date: Thu, 9 May 2013 16:00:30 -0500 Subject: [PATCH 10/16] Documenting classes! --- docs/tutorial.rst | 40 ++++++++++++++++++++++++++++++++++++++++ 1 file changed, 40 insertions(+) diff --git a/docs/tutorial.rst b/docs/tutorial.rst index 99f18fc..75df541 100644 --- a/docs/tutorial.rst +++ b/docs/tutorial.rst @@ -433,6 +433,46 @@ The Hy equivalent: (import pprint) (pprint.pprint (, foo bar args kwargs))) +Finally, of course we need classes! In python we might have a class +like:: + + class FooBar (object): + def __init__(self, x): + self.x = x + + def get_x(self): + return self.x + + +In Hy: + +.. code-block:: clj + + (defclass FooBar [object] + [[--init-- + (fn [self x] + (setv self.x x))] + + [get-x + (fn [self] + self.x)]]) + + +You can also do class-level attributes. In Python:: + + class Customer(models.Model): + name = models.CharField(max_length=255) + address = models.TextField() + notes = models.TextField() + +In Hy: + +.. code-block:: clj + + (defclass Customer [models.Model] + [[name (kwapply (models.CharField) {"max_length" 255})] + [address (models.TextField)] + [notes (models.TextField)]]) Protips! From b68d5ac3a33d12fb7686359dc6181b97926431db Mon Sep 17 00:00:00 2001 From: Christopher Allan Webber Date: Thu, 9 May 2013 16:01:07 -0500 Subject: [PATCH 11/16] Remove document classes TODO --- docs/tutorial.rst | 1 - 1 file changed, 1 deletion(-) diff --git a/docs/tutorial.rst b/docs/tutorial.rst index 75df541..3d6d9e0 100644 --- a/docs/tutorial.rst +++ b/docs/tutorial.rst @@ -521,7 +521,6 @@ TODO - How do I index into arrays or dictionaries? - How do I do array ranges? e.g. x[5:] or y[2:10] - - How do I define classes? - Blow your mind with macros! - Where's my banana??? - Mention that you can import .hy files in .py files and vice versa! From c5dbc39ee155b8079cfba57f6273be4ec47dc798 Mon Sep 17 00:00:00 2001 From: Christopher Allan Webber Date: Thu, 9 May 2013 16:02:25 -0500 Subject: [PATCH 12/16] Fixing documentation error with extra double-colon --- docs/tutorial.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/tutorial.rst b/docs/tutorial.rst index 3d6d9e0..1fcb3e5 100644 --- a/docs/tutorial.rst +++ b/docs/tutorial.rst @@ -409,7 +409,7 @@ The same thing in Hy:: See how we use kwapply to handle the fancy pssing? :) There's also a dictionary-style keyword arguments construction that -looks like:: +looks like: .. code-block:: clj From 204bc9c39e1c6b843d3e2ea26d88d5d3ecc985f2 Mon Sep 17 00:00:00 2001 From: Christopher Allan Webber Date: Thu, 9 May 2013 16:04:12 -0500 Subject: [PATCH 13/16] Fixing *args and **kwargs notation in docs --- docs/tutorial.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/tutorial.rst b/docs/tutorial.rst index 1fcb3e5..9b3127d 100644 --- a/docs/tutorial.rst +++ b/docs/tutorial.rst @@ -419,7 +419,7 @@ looks like: The difference here is that since it's a dictionary, you can't rely on any specific ordering to the arguments. -Hy also supports **args and **kwargs. In Python:: +Hy also supports ``*args`` and ``**kwargs``. In Python:: def some_func(foo, bar, *args, **kwargs): import pprint From 0571bab3820477b7f2e419d9872d5154922b2d26 Mon Sep 17 00:00:00 2001 From: Christopher Allan Webber Date: Thu, 9 May 2013 16:08:39 -0500 Subject: [PATCH 14/16] Return None in --init-- because due to a syntax flaw we have to! --- docs/tutorial.rst | 5 ++++- 1 file changed, 4 insertions(+), 1 deletion(-) diff --git a/docs/tutorial.rst b/docs/tutorial.rst index 9b3127d..cad438f 100644 --- a/docs/tutorial.rst +++ b/docs/tutorial.rst @@ -451,7 +451,10 @@ In Hy: (defclass FooBar [object] [[--init-- (fn [self x] - (setv self.x x))] + (setv self.x x) + ; Currently needed for --init-- because __init__ needs None + ; Hopefully this will go away :) + None)] [get-x (fn [self] From c4ca00a485e1b316a40416e994ec7cbe2b69fa59 Mon Sep 17 00:00:00 2001 From: Nicolas Dandrimont Date: Thu, 9 May 2013 23:42:22 +0200 Subject: [PATCH 15/16] Fix the PDF docs build --- docs/conf.py | 6 ++++++ 1 file changed, 6 insertions(+) diff --git a/docs/conf.py b/docs/conf.py index f0932da..40eb062 100644 --- a/docs/conf.py +++ b/docs/conf.py @@ -181,6 +181,12 @@ latex_elements = { # Additional stuff for the LaTeX preamble. #'preamble': '', + +# Stuff for unicode characters + 'utf8extra': r''' +\DeclareUnicodeCharacter{2698}{FLOWER} +\DeclareUnicodeCharacter{2665}{HEART} +''', } # Grouping the document tree into LaTeX files. List of tuples From 938d783e3b57233d80908ba59a0f64990eb7eb39 Mon Sep 17 00:00:00 2001 From: Nicolas Dandrimont Date: Thu, 9 May 2013 23:42:37 +0200 Subject: [PATCH 16/16] Symbols get mangled without double-underscores now --- docs/language/api.rst | 4 ++-- 1 file changed, 2 insertions(+), 2 deletions(-) diff --git a/docs/language/api.rst b/docs/language/api.rst index 8f63838..47d99d8 100644 --- a/docs/language/api.rst +++ b/docs/language/api.rst @@ -24,8 +24,8 @@ languages. * UTF-8 entities will be encoded using `punycode `_ and prefixed with - `__hy_`. For instance, `⚘` will become `__hy_w7h`, and `♥` will become - `__hy_g6h`. + `hy_`. For instance, `⚘` will become `hy_w7h`, and `♥` will become + `hy_g6h`. * Symbols that contain dashes will have them replaced with underscores. For example, `render-template` will become `render_template`.