From 7c36fe9a145bcabf48ecbb6eaa8962fba625ab6d Mon Sep 17 00:00:00 2001 From: Tuukka Turto Date: Sat, 3 Oct 2015 13:24:43 +0300 Subject: [PATCH 01/25] Add section about macros in tutorial relates #926 --- docs/tutorial.rst | 58 +++++++++++++++++++++++++++++++++++++++++++++++ 1 file changed, 58 insertions(+) diff --git a/docs/tutorial.rst b/docs/tutorial.rst index fc5e3ff..cf76538 100644 --- a/docs/tutorial.rst +++ b/docs/tutorial.rst @@ -503,6 +503,64 @@ In Hy: address (models.TextField) notes (models.TextField)]) +Macros +====== + +One really powerful feature of Hy are macros. They are small functios that are +used to generate code (or data). When program written in Hy is started, the +macros are executed and their output is placed in program source. After this, +the program starts executing normally. Very simple example: + +.. code-block:: clj + + => (defmacro hello [person] + ... `(print "Hello there," ~person)) + => (Hello "Tuukka") + Hello there, Tuukka + +The thing to notice here is that hello macro doesn't output anything on +screen. Instead it creates piece of code that is then executed and prints on +screen. Macro writes a piece of program that looks like this (provided that +we used "Tuukka" as parameter: + +.. code-block:: clj + + (print "Hello there," Tuukka) + +We can also manipulate code with macros: + +.. code-block:: clj + + => (defmacro rev [code] + ... (let [op (last code) params (list (butlast code))] + ... `(~op ~@params))) + => (rev (1 2 3 +)) + 6 + +The code that was generated with this macro just switched around some the +elements, so by the time program started executing, it actually red: + +.. code-block:: clj + + (+ 1 2 3) + +Sometimes it's nice to have a very short name for macro that doesn't take much +space or use extra parentheses. Reader macros can be pretty useful in these +situations (and since Hy operates well with unicode, we aren't running out of +characters that soon): + +.. code-block:: clj + + => (defreader ↻ [code] + ... (let [op (last code) params (list (butlast code))] + ... `(~op ~@params))) + => #↻(1 2 3 +) + 6 + +Macros are useful when one wished to extend the Hy or write their own +language on top of that. Many features of Hy are macros, like ``when``, +``cond`` and ``->``. + Hy <-> Python interop ===================== From 9e5f881958bdfd72484e2eeb147926ed6dc00587 Mon Sep 17 00:00:00 2001 From: Ryan Gonzalez Date: Tue, 13 Oct 2015 16:48:23 -0500 Subject: [PATCH 02/25] Give an error when &kwonly or &kwargs is used in a macro definition (closes #959) --- hy/compiler.py | 3 +++ tests/native_tests/native_macros.hy | 16 ++++++++++++++++ 2 files changed, 19 insertions(+) diff --git a/hy/compiler.py b/hy/compiler.py index 4574a3f..1a2a4f0 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -2371,6 +2371,9 @@ class HyASTCompiler(object): raise HyTypeError(name, ("received a `%s' instead of a symbol " "for macro name" % type(name).__name__)) name = HyString(name).replace(name) + for kw in ("&kwonly", "&kwargs"): + if kw in expression[0]: + raise HyTypeError(name, "macros cannot use %s" % kw) new_expression = HyExpression([ HySymbol("with_decorator"), HyExpression([HySymbol("hy.macros.macro"), name]), diff --git a/tests/native_tests/native_macros.hy b/tests/native_tests/native_macros.hy index 0d10609..5634c98 100644 --- a/tests/native_tests/native_macros.hy +++ b/tests/native_tests/native_macros.hy @@ -1,3 +1,5 @@ +(import [hy.errors [HyTypeError]]) + (defmacro rev [&rest body] "Execute the `body` statements in reverse" (quasiquote (do (unquote-splice (list (reversed body)))))) @@ -48,6 +50,20 @@ (defmacro bar [x y] (foo x y)) +(defn test-macro-kw [] + "NATIVE: test that an error is raised when &kwonly or &kwargs is used in a macro" + (try + (eval '(defmacro f [&kwonly a b])) + (except [e HyTypeError] + (assert (= e.message "macros cannot use &kwonly"))) + (else (assert false))) + + (try + (eval '(defmacro f [&kwargs kw])) + (except [e HyTypeError] + (assert (= e.message "macros cannot use &kwargs"))) + (else (assert false)))) + (defn test-fn-calling-macro [] "NATIVE: test macro calling a plain function" (assert (= 3 (bar 1 2)))) From 7e13bb4e5f142d2fbfaa03f88ba1263978adc930 Mon Sep 17 00:00:00 2001 From: Matthew Egan Odendahl Date: Fri, 15 Apr 2016 20:33:42 -0600 Subject: [PATCH 03/25] add as-> macro --- hy/core/macros.hy | 9 +++++++++ 1 file changed, 9 insertions(+) diff --git a/hy/core/macros.hy b/hy/core/macros.hy index d39aabe..dc3c316 100644 --- a/hy/core/macros.hy +++ b/hy/core/macros.hy @@ -30,6 +30,15 @@ [hy.models.symbol [HySymbol]] [hy._compat [PY33 PY34]]) +(defmacro as-> [head name &rest rest] + "Becomes ordered assignments to the provided name. The previous result + is thus available in each form. Returns the final result, and leaves the name + bound to it in the local scope. This behaves like the threading macros, but + requires you to specify the threading point per form via the name." + `(do (setv + ~name ~head + ~@(interleave (repeat name) rest)) + ~name)) (defmacro with [args &rest body] "shorthand for nested with* loops: From e0ecaca315cdc061905e1464e5245e1c13a96f40 Mon Sep 17 00:00:00 2001 From: Matthew Egan Odendahl Date: Sun, 19 Jun 2016 21:14:16 -0600 Subject: [PATCH 04/25] Clarify as-> docstring --- hy/core/macros.hy | 9 +++++---- 1 file changed, 5 insertions(+), 4 deletions(-) diff --git a/hy/core/macros.hy b/hy/core/macros.hy index dc3c316..e6c8e16 100644 --- a/hy/core/macros.hy +++ b/hy/core/macros.hy @@ -31,10 +31,11 @@ [hy._compat [PY33 PY34]]) (defmacro as-> [head name &rest rest] - "Becomes ordered assignments to the provided name. The previous result - is thus available in each form. Returns the final result, and leaves the name - bound to it in the local scope. This behaves like the threading macros, but - requires you to specify the threading point per form via the name." + "Expands to sequence of assignments to the provided name, starting with head. + The previous result is thus available in the subsequent form. Returns the + final result, and leaves the name bound to it in the local scope. This behaves + much like the other threading macros, but requires you to specify the threading + point per form via the name instead of always the first or last arument." `(do (setv ~name ~head ~@(interleave (repeat name) rest)) From d384580de1a427f990b34dcccb42ce43e63adc6a Mon Sep 17 00:00:00 2001 From: Ryan Gonzalez Date: Thu, 7 Jul 2016 11:24:04 -0500 Subject: [PATCH 05/25] Disallow &key in macros --- hy/compiler.py | 2 +- tests/native_tests/native_macros.hy | 8 +++++++- 2 files changed, 8 insertions(+), 2 deletions(-) diff --git a/hy/compiler.py b/hy/compiler.py index 99c36d1..677dab4 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -2428,7 +2428,7 @@ class HyASTCompiler(object): raise HyTypeError(name, ("received a `%s' instead of a symbol " "for macro name" % type(name).__name__)) name = HyString(name).replace(name) - for kw in ("&kwonly", "&kwargs"): + for kw in ("&kwonly", "&kwargs", "&key"): if kw in expression[0]: raise HyTypeError(name, "macros cannot use %s" % kw) new_expression = HyExpression([ diff --git a/tests/native_tests/native_macros.hy b/tests/native_tests/native_macros.hy index 8c2f602..eca4538 100644 --- a/tests/native_tests/native_macros.hy +++ b/tests/native_tests/native_macros.hy @@ -51,7 +51,7 @@ (foo x y)) (defn test-macro-kw [] - "NATIVE: test that an error is raised when &kwonly or &kwargs is used in a macro" + "NATIVE: test that an error is raised when &kwonly, &kwargs, or &key is used in a macro" (try (eval '(defmacro f [&kwonly a b])) (except [e HyTypeError] @@ -62,6 +62,12 @@ (eval '(defmacro f [&kwargs kw])) (except [e HyTypeError] (assert (= e.message "macros cannot use &kwargs"))) + (else (assert false))) + + (try + (eval '(defmacro f [&key {"kw" "xyz"}])) + (except [e HyTypeError] + (assert (= e.message "macros cannot use &key"))) (else (assert false)))) (defn test-fn-calling-macro [] From f1684e63068658947577cae96133b436cd9bccc9 Mon Sep 17 00:00:00 2001 From: Pierre-Yves Date: Sat, 6 Aug 2016 20:25:07 -0500 Subject: [PATCH 06/25] Add doc and related link to good-first-bug tag (#1086) Fixes #1031 --- CONTRIBUTING.rst | 7 +++++-- 1 file changed, 5 insertions(+), 2 deletions(-) diff --git a/CONTRIBUTING.rst b/CONTRIBUTING.rst index f9c2ec5..cf3ddf6 100644 --- a/CONTRIBUTING.rst +++ b/CONTRIBUTING.rst @@ -7,7 +7,9 @@ helps in making Hy more awesome. Pull requests are great! We love them; here is a quick guide: - `Fork the repo`_ and create a topic branch for a feature/fix. Avoid - making changes directly on the master branch. + making changes directly on the master branch. If you would like to + contribute but don't know how to begin, the `good-first-bug`_ label + of the `issue tracker`_ is the place to go. (If you're new to Git: `Start Here`_) - All incoming features should be accompanied with tests. @@ -93,4 +95,5 @@ http://contributor-covenant.org/version/1/1/0/. .. _Contributor Covenant: http://contributor-covenant.org .. _issue tracker: https://github.com/hylang/hy/issues .. _Fork the Repo: https://help.github.com/articles/fork-a-repo/ -.. _Start Here: http://rogerdudler.github.io/git-guide/) \ No newline at end of file +.. _Start Here: http://rogerdudler.github.io/git-guide/) +.. _good-first-bug: http://github.com/hylang/hy/issues?q=is%3Aissue+is%3Aopen+label%3Agood-first-bug From 6a452d70a46f00fa521cc543e321d8efa391baa8 Mon Sep 17 00:00:00 2001 From: Michael Hunsinger Date: Sat, 20 Aug 2016 02:35:02 -0600 Subject: [PATCH 07/25] Added documentation for walk, postwalk and prewalk --- docs/contrib/index.rst | 5 ++- docs/contrib/walk.rst | 94 ++++++++++++++++++++++++++++++++++++++++++ 2 files changed, 97 insertions(+), 2 deletions(-) create mode 100644 docs/contrib/walk.rst diff --git a/docs/contrib/index.rst b/docs/contrib/index.rst index 612ccdd..ed75add 100644 --- a/docs/contrib/index.rst +++ b/docs/contrib/index.rst @@ -7,8 +7,9 @@ Contents: .. toctree:: :maxdepth: 3 + alias anaphoric + flow loop multi - flow - alias + walk diff --git a/docs/contrib/walk.rst b/docs/contrib/walk.rst new file mode 100644 index 0000000..01e2bc0 --- /dev/null +++ b/docs/contrib/walk.rst @@ -0,0 +1,94 @@ +==== +walk +==== + +.. versionadded:: 0.11.0 + +Functions +========= + +.. _walk: + +walk +----- + +Usage: `(walk inner outer form)` + +``walk`` traverses ``form``, an arbitrary data structure. Applies +``inner`` to each element of form, building up a data structure of the +same type. Applies ``outer`` to the result. + +Example: + +.. code-block:: hy + + => (import [hy.contrib.walk [walk]]) + => (setv a '(a b c d e f)) + => (walk ord identity a) + (97 98 99 100 101 102) + => (walk ord first a) + 97 + +postwalk +--------- + +.. _postwalk: + +Usage: `(postwalk f form)` + +Performs depth-first, post-order traversal of ``form``. Calls ``f`` on +each sub-form, uses ``f`` 's return value in place of the original. + +.. code-block:: hy + + => (import [hy.contrib.walk [postwalk]]) + => (def trail '([1 2 3] [4 [5 6 [7]]])) + => (defn walking [x] + (print "Walking:" x) + x ) + => (postwalk walking trail) + Walking: 1 + Walking: 2 + Walking: 3 + Walking: (1 2 3) + Walking: 4 + Walking: 5 + Walking: 6 + Walking: 7 + Walking: (7) + Walking: (5 6 [7]) + Walking: (4 [5 6 [7]]) + Walking: ([1 2 3] [4 [5 6 [7]]]) + ([1 2 3] [4 [5 6 [7]]]) + +prewalk +-------- + +.. _prewalk: + +Usage: `(prewalk f form)` + +Performs depth-first, pre-order traversal of ``form``. Calls ``f`` on +each sub-form, uses ``f`` 's return value in place of the original. + +.. code-block:: hy + + => (import [hy.contrib.walk [prewalk]]) + => (def trail '([1 2 3] [4 [5 6 [7]]])) + => (defn walking [x] + (print "Walking:" x) + x ) + => (prewalk walking trail) + Walking: ([1 2 3] [4 [5 6 [7]]]) + Walking: [1 2 3] + Walking: 1 + Walking: 2 + Walking: 3 + Walking: [4 [5 6 [7]]] + Walking: 4 + Walking: [5 6 [7]] + Walking: 5 + Walking: 6 + Walking: [7] + Walking: 7 + ([1 2 3] [4 [5 6 [7]]]) From b94211251c647a5622e0dcf35f2f19a8bb799882 Mon Sep 17 00:00:00 2001 From: Jakub Wilk Date: Sun, 4 Sep 2016 22:35:46 +0200 Subject: [PATCH 08/25] Use HTTPS for Wikipedia URLs --- docs/contrib/anaphoric.rst | 2 +- docs/contrib/loop.rst | 2 +- docs/language/api.rst | 2 +- docs/language/internals.rst | 2 +- 4 files changed, 4 insertions(+), 4 deletions(-) diff --git a/docs/contrib/anaphoric.rst b/docs/contrib/anaphoric.rst index cce39e0..51ece00 100644 --- a/docs/contrib/anaphoric.rst +++ b/docs/contrib/anaphoric.rst @@ -11,7 +11,7 @@ concise and easy to read. deliberately captures some form supplied to the macro which may be referred to by an anaphor (an expression referring to another). - -- Wikipedia (http://en.wikipedia.org/wiki/Anaphoric_macro) + -- Wikipedia (https://en.wikipedia.org/wiki/Anaphoric_macro) To use these macros you need to require the hy.contrib.anaphoric module like so: diff --git a/docs/contrib/loop.rst b/docs/contrib/loop.rst index 9968aab..9ff80a2 100644 --- a/docs/contrib/loop.rst +++ b/docs/contrib/loop.rst @@ -24,7 +24,7 @@ tail-call optimization (TCO) in their Hy code. position to be implemented as efficiently as goto statements, thus allowing efficient structured programming. - -- Wikipedia (http://en.wikipedia.org/wiki/Tail_call) + -- Wikipedia (https://en.wikipedia.org/wiki/Tail_call) Macros ====== diff --git a/docs/language/api.rst b/docs/language/api.rst index c96e0b6..7923cee 100644 --- a/docs/language/api.rst +++ b/docs/language/api.rst @@ -22,7 +22,7 @@ languages. string. For example, ``foo`` will become ``FOO``. * UTF-8 entities will be encoded using - `punycode `_ and prefixed with + `punycode `_ and prefixed with ``hy_``. For instance, ``⚘`` will become ``hy_w7h``, ``♥`` will become ``hy_g6h``, and ``i♥u`` will become ``hy_iu_t0x``. diff --git a/docs/language/internals.rst b/docs/language/internals.rst index d543847..c4b9db6 100644 --- a/docs/language/internals.rst +++ b/docs/language/internals.rst @@ -170,7 +170,7 @@ Cons Cells cells`_. Cons cells are especially useful to mimic features of "usual" LISP variants such as Scheme or Common Lisp. -.. _cons cells: http://en.wikipedia.org/wiki/Cons +.. _cons cells: https://en.wikipedia.org/wiki/Cons A cons cell is a 2-item object, containing a ``car`` (head) and a ``cdr`` (tail). In some Lisp variants, the cons cell is the fundamental From a60b749d3eaae764648a498297b2afad069e3a00 Mon Sep 17 00:00:00 2001 From: Kodi Arfer Date: Tue, 20 Sep 2016 13:05:52 -0700 Subject: [PATCH 09/25] Make unary + call __pos__ Fixes #1109. __pos__ doesn't work on strings, lists, or tuples, so I've removed some tests that required (= (+ "a") "a") etc. --- hy/compiler.py | 28 ++++++++++++++++++++-------- hy/core/shadow.hy | 2 +- tests/native_tests/mathematics.hy | 12 ++++++++++++ tests/native_tests/shadow.hy | 4 ---- 4 files changed, 33 insertions(+), 13 deletions(-) diff --git a/hy/compiler.py b/hy/compiler.py index ca8c0e9..d663d58 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -1910,7 +1910,6 @@ class HyASTCompiler(object): col_offset=child.start_column) return ret - @builds("+") @builds("*") @builds("/") @builds("//") @@ -1918,8 +1917,7 @@ class HyASTCompiler(object): if len(expression) > 2: return self.compile_maths_expression(expression) else: - id_op = {"+": HyInteger(0), "*": HyInteger(1), "/": HyInteger(1), - "//": HyInteger(1)} + id_op = {"*": HyInteger(1), "/": HyInteger(1), "//": HyInteger(1)} op = expression.pop(0) arg = expression.pop(0) if expression else id_op[op] @@ -1930,20 +1928,34 @@ class HyASTCompiler(object): ]).replace(expression) return self.compile_maths_expression(expr) - @builds("-") - @checkargs(min=1) - def compile_maths_expression_sub(self, expression): + def compile_maths_expression_additive(self, expression): if len(expression) > 2: return self.compile_maths_expression(expression) else: - arg = expression[1] + op = {"+": ast.UAdd, "-": ast.USub}[expression.pop(0)]() + arg = expression.pop(0) ret = self.compile(arg) - ret += ast.UnaryOp(op=ast.USub(), + ret += ast.UnaryOp(op=op, operand=ret.force_expr, lineno=arg.start_line, col_offset=arg.start_column) return ret + @builds("+") + def compile_maths_expression_add(self, expression): + if len(expression) == 1: + # Nullary + + return ast.Num(n=long_type(0), + lineno=expression.start_line, + col_offset=expression.start_column) + else: + return self.compile_maths_expression_additive(expression) + + @builds("-") + @checkargs(min=1) + def compile_maths_expression_sub(self, expression): + return self.compile_maths_expression_additive(expression) + @builds("+=") @builds("/=") @builds("//=") diff --git a/hy/core/shadow.hy b/hy/core/shadow.hy index b63219d..e66d075 100644 --- a/hy/core/shadow.hy +++ b/hy/core/shadow.hy @@ -30,7 +30,7 @@ (if (zero? count) (raise (TypeError "Need at least 1 argument to add/concatenate")) (if (= count 1) - (get args 0) + (operator.pos (get args 0)) (reduce operator.add args))))) diff --git a/tests/native_tests/mathematics.hy b/tests/native_tests/mathematics.hy index 58c5457..5ab90c4 100644 --- a/tests/native_tests/mathematics.hy +++ b/tests/native_tests/mathematics.hy @@ -28,6 +28,18 @@ (assert (= 0 (+))))) +(defn test-add-unary [] + "NATIVE: test that unary + calls __pos__" + + (defclass X [object] + [__pos__ (fn [self] "called __pos__")]) + (assert (= (+ (X)) "called __pos__")) + + ; Make sure the shadowed version works, too. + (setv f +) + (assert (= (f (X)) "called __pos__"))) + + (setv test_div (fn [] "NATIVE: Test division" (assert (= 25 (/ 100 2 2))) diff --git a/tests/native_tests/shadow.hy b/tests/native_tests/shadow.hy index 0ca08ba..ce578bf 100644 --- a/tests/native_tests/shadow.hy +++ b/tests/native_tests/shadow.hy @@ -8,13 +8,9 @@ (assert (= (x 1 2 3 4) 10)) (assert (= (x 1 2 3 4 5) 15)) ; with strings - (assert (= (x "a") - "a")) (assert (= (x "a" "b" "c") "abc")) ; with lists - (assert (= (x ["a"]) - ["a"])) (assert (= (x ["a"] ["b"] ["c"]) ["a" "b" "c"])))) From 4db667322c1869342704805dcb8b64bb4b2f7a97 Mon Sep 17 00:00:00 2001 From: Jakukyo Friel Date: Thu, 22 Sep 2016 08:32:35 +0800 Subject: [PATCH 10/25] Doc: Fix typo (missing quote) (#1096) --- docs/tutorial.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/tutorial.rst b/docs/tutorial.rst index 7f9b4eb..834332d 100644 --- a/docs/tutorial.rst +++ b/docs/tutorial.rst @@ -292,7 +292,7 @@ You can do the following: (if (try-some-thing) (do (print "this is if true") - (print "and why not, let's keep talking about how true it is!)) + (print "and why not, let's keep talking about how true it is!")) (print "this one's still simply just false")) You can see that we used ``do`` to wrap multiple statements. If you're From 35de9988714656625429b91da076473057e48e21 Mon Sep 17 00:00:00 2001 From: Jakukyo Friel Date: Thu, 22 Sep 2016 08:33:32 +0800 Subject: [PATCH 11/25] Docs: Fix rst markup in docs/contrib/loop.rst. (#1098) In RestrutruedText, double backtick quotes (instead of single backtick quotes in markdown) are used for code span. --- docs/contrib/loop.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/contrib/loop.rst b/docs/contrib/loop.rst index 9ff80a2..aaef239 100644 --- a/docs/contrib/loop.rst +++ b/docs/contrib/loop.rst @@ -39,7 +39,7 @@ rebinds the variables set in the recursion point and sends code execution back to that recursion point. If ``recur`` is used in a non-tail position, an exception is raised. -Usage: `(loop bindings &rest body)` +Usage: ``(loop bindings &rest body)`` Example: From 53353c58f2fadc1063eb75110e94e14d244be986 Mon Sep 17 00:00:00 2001 From: "Zack M. Davis" Date: Wed, 21 Sep 2016 22:05:53 -0700 Subject: [PATCH 12/25] fix Hy on recent Pythons In issue #1111, @tianon reported that Hy didn't work with Python 3.6.0b1: trying to evaluate a simple expression at the REPL blew up with `TypeError: required field "is_async" missing from comprehension`. This was due to a grammar change (https://www.python.org/dev/peps/pep-0530/#grammar-updates) in the implementation (https://hg.python.org/cpython/rev/cf91a929b81a) of PEP 530, which we can easily accomodate. --- hy/compiler.py | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/hy/compiler.py b/hy/compiler.py index ca8c0e9..6fc1551 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -1516,7 +1516,8 @@ class HyASTCompiler(object): gen.append(ast.comprehension( target=target, iter=gen_res.force_expr, - ifs=[])) + ifs=[], + is_async=False)) if cond.expr: gen[-1].ifs.append(cond.expr) From d81b71f2423b7fbfe38bc5ed27583c7d56cfe243 Mon Sep 17 00:00:00 2001 From: Jakukyo Friel Date: Thu, 22 Sep 2016 20:20:26 +0800 Subject: [PATCH 13/25] Doc: CLI: mention `--spy` only works in REPL mode. (#1097) --- docs/language/cli.rst | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/docs/language/cli.rst b/docs/language/cli.rst index 28bd067..ae9b620 100644 --- a/docs/language/cli.rst +++ b/docs/language/cli.rst @@ -35,7 +35,7 @@ Command Line Options .. cmdoption:: --spy - Print equivalent Python code before executing. For example:: + Print equivalent Python code before executing in REPL. For example:: => (defn salutationsnm [name] (print (+ "Hy " name "!"))) def salutationsnm(name): @@ -45,6 +45,7 @@ Command Line Options Hy YourName! => + `--spy` only works on REPL mode. .. versionadded:: 0.9.11 .. cmdoption:: --show-tracebacks From d4494151f4dcb454e15a297e3012c68b356ffb35 Mon Sep 17 00:00:00 2001 From: "Zack M. Davis" Date: Thu, 22 Sep 2016 23:24:14 -0700 Subject: [PATCH 14/25] experiment: can we convince Travis CI to test with nightly Python? The docs say that "nightly" is supported, but it's less clear exactly what our .travis.yml should say to make this happen. --- .travis.yml | 2 ++ tox.ini | 2 +- 2 files changed, 3 insertions(+), 1 deletion(-) diff --git a/.travis.yml b/.travis.yml index 9f8dea7..8ab3a44 100644 --- a/.travis.yml +++ b/.travis.yml @@ -4,6 +4,8 @@ matrix: include: - python: 3.5 env: TOXENV=py35 + - python: nightly + env: TOXENV=py36 env: - TOXENV=py27 - TOXENV=py33 diff --git a/tox.ini b/tox.ini index 2bea02d..0162794 100644 --- a/tox.ini +++ b/tox.ini @@ -1,5 +1,5 @@ [tox] -envlist = py27,pypy,py33,py34,py35,flake8 +envlist = py27,pypy,py33,py34,py35,py36,flake8 skipsdist = True [testenv] From 1d3142c8de003f037a674e36e847ee81402319f8 Mon Sep 17 00:00:00 2001 From: gilch Date: Sat, 24 Sep 2016 15:54:32 -0600 Subject: [PATCH 15/25] change quickstart install from PyPI to GitHub (#1116) * change Quickstart install from PyPI to GitHub --- docs/quickstart.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/docs/quickstart.rst b/docs/quickstart.rst index 24bca21..7ee72e0 100644 --- a/docs/quickstart.rst +++ b/docs/quickstart.rst @@ -13,7 +13,7 @@ Quickstart 1. Create a `Virtual Python Environment `_. 2. Activate your Virtual Python Environment. -3. Install `hy from PyPI `_ with ``pip install hy``. +3. Install `hy from GitHub `_ with ``$ pip install git+https://github.com/hylang/hy.git``. 4. Start a REPL with ``hy``. 5. Type stuff in the REPL:: From 399d2bcf2116fdf0a453ed1cc1192e994b9d4a90 Mon Sep 17 00:00:00 2001 From: gilch Date: Sun, 25 Sep 2016 21:29:53 -0600 Subject: [PATCH 16/25] clarify documentation of import (#1121) * clarify documentation of import Add a comment showing the equivalent python of the long example, and demonstrate how to get `from` and `as` on the same line. * add python example comment in last import example --- docs/language/api.rst | 11 ++++++++++- 1 file changed, 10 insertions(+), 1 deletion(-) diff --git a/docs/language/api.rst b/docs/language/api.rst index 7923cee..fa09c82 100644 --- a/docs/language/api.rst +++ b/docs/language/api.rst @@ -953,11 +953,20 @@ that ``import`` can be used. (import [sys :as systest]) ;; You can list as many imports as you like of different types. + ;; + ;; Python: + ;; from tests.resources import kwtest, function_with_a_dash + ;; from os.path import exists, isdir as is_dir, isfile as is_file + ;; import sys as systest (import [tests.resources [kwtest function-with-a-dash]] - [os.path [exists isdir isfile]] + [os.path [exists + isdir :as dir? + isfile :as file?]] [sys :as systest]) ;; Import all module functions into current namespace + ;; + ;; Python: from sys import * (import [sys [*]]) From 0880610401ae1249cd434514ddc607a1f25705d8 Mon Sep 17 00:00:00 2001 From: Kodi Arfer Date: Mon, 26 Sep 2016 09:47:04 -0700 Subject: [PATCH 17/25] Don't sort or deduplicate the items in a HySet Fixes #1120. I also added hy.models._wrapper[set] so a macro can return an ordinary set in place of a HySet. --- hy/models/set.py | 11 ++++------- tests/lex/test_lex.py | 10 ++++++++++ tests/models/test_set.py | 2 +- tests/native_tests/language.hy | 1 + tests/native_tests/native_macros.hy | 3 +++ 5 files changed, 19 insertions(+), 8 deletions(-) diff --git a/hy/models/set.py b/hy/models/set.py index 3a81aed..fda7b52 100644 --- a/hy/models/set.py +++ b/hy/models/set.py @@ -18,19 +18,16 @@ # FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER # DEALINGS IN THE SOFTWARE. +from hy.models import _wrappers, wrap_value from hy.models.list import HyList -from functools import reduce class HySet(HyList): """ - Hy set (actually a list that pretends to be a set) + Hy set (just a representation of a set) """ - def __init__(self, items): - items = sorted(items) - items = list(reduce(lambda r, v: v in r and r or r+[v], items, [])) - super(HySet, self).__init__(items) - def __repr__(self): return "#{%s}" % (" ".join([repr(x) for x in self])) + +_wrappers[set] = lambda s: HySet(wrap_value(x) for x in s) diff --git a/tests/lex/test_lex.py b/tests/lex/test_lex.py index 4413a6b..81407cc 100644 --- a/tests/lex/test_lex.py +++ b/tests/lex/test_lex.py @@ -230,6 +230,16 @@ def test_sets(): HyExpression([HySymbol("baz"), HySymbol("quux")]) ])] + # Duplicate items in a literal set should be okay (and should + # be preserved). + objs = tokenize("#{1 2 1 1 2 1}") + assert objs == [HySet([HyInteger(n) for n in [1, 2, 1, 1, 2, 1]])] + assert len(objs[0]) == 6 + + # https://github.com/hylang/hy/issues/1120 + objs = tokenize("#{a 1}") + assert objs == [HySet([HySymbol("a"), HyInteger(1)])] + def test_nospace(): """ Ensure we can tokenize without spaces if we have to """ diff --git a/tests/models/test_set.py b/tests/models/test_set.py index ceea726..0462c2c 100644 --- a/tests/models/test_set.py +++ b/tests/models/test_set.py @@ -5,4 +5,4 @@ hyset = HySet([3, 1, 2, 2]) def test_set(): - assert hyset == [1, 2, 3] + assert hyset == [3, 1, 2, 2] diff --git a/tests/native_tests/language.hy b/tests/native_tests/language.hy index 0300ffb..54b3e61 100644 --- a/tests/native_tests/language.hy +++ b/tests/native_tests/language.hy @@ -47,6 +47,7 @@ (defn test-sets [] "NATIVE: test sets work right" (assert (= #{1 2 3 4} (| #{1 2} #{3 4}))) + (assert (= (type #{1 2 3 4}) set)) (assert (= #{} (set)))) diff --git a/tests/native_tests/native_macros.hy b/tests/native_tests/native_macros.hy index eca4538..788e52b 100644 --- a/tests/native_tests/native_macros.hy +++ b/tests/native_tests/native_macros.hy @@ -39,6 +39,9 @@ (defmacro a-dict [] {1 2}) (assert (= (a-dict) {1 2})) +(defmacro a-set [] #{1 2}) +(assert (= (a-set) #{1 2})) + (defmacro a-none []) (assert (= (a-none) None)) From a33f80b2ce45a0c2f633ab13dce79852ee6dfdbf Mon Sep 17 00:00:00 2001 From: Kodi Arfer Date: Thu, 29 Sep 2016 09:18:04 -0700 Subject: [PATCH 18/25] Allow --version as a synonym for -v --version is a GNU standard that is also common for non-GNU programs, such as Python. --- hy/cmdline.py | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/hy/cmdline.py b/hy/cmdline.py index 195a5bc..35f75c6 100644 --- a/hy/cmdline.py +++ b/hy/cmdline.py @@ -274,7 +274,7 @@ def cmdline_handler(scriptname, argv): parser.add_argument("--spy", action="store_true", help="print equivalent Python code before executing") - parser.add_argument("-v", action="version", version=VERSION) + parser.add_argument("-v", "--version", action="version", version=VERSION) parser.add_argument("--show-tracebacks", action="store_true", help="show complete tracebacks for Hy exceptions") From 1711b1f5ad303a979df9bcb8d4e8b8cc9aea571d Mon Sep 17 00:00:00 2001 From: Bheesham Persaud Date: Tue, 4 Oct 2016 21:00:19 -0400 Subject: [PATCH 19/25] Add PHONY targets; wrap call to recursive make; add path to find. I've marked added the rest of the phony build targets; wrapped the recursive make call[0], and added the path specifier to the find command for compatibility with BSD find. [0]: https://www.gnu.org/software/make/manual/html_node/Recursion.html --- Makefile | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/Makefile b/Makefile index ccf8456..40d8eaf 100644 --- a/Makefile +++ b/Makefile @@ -23,7 +23,7 @@ all: @echo "" docs: - make -C docs html + $(MAKE) -C docs html upload: r python setup.py sdist upload @@ -72,10 +72,10 @@ coveralls: clean: @find . -name "*.pyc" -exec rm {} \; - @find -name __pycache__ -delete + @find . -name __pycache__ -delete @${RM} -r -f .tox @${RM} -r -f dist @${RM} -r -f *.egg-info @${RM} -r -f docs/_build -.PHONY: docs +.PHONY: all docs upload full venv dev test tox flake clear d diff r python coveralls clean From 72ded830b61ebc060dc7c6d2c1d4bcde9ef796e2 Mon Sep 17 00:00:00 2001 From: Jorge Date: Sun, 9 Oct 2016 13:01:08 +0200 Subject: [PATCH 20/25] Remove invalid `)` from URL --- CONTRIBUTING.rst | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/CONTRIBUTING.rst b/CONTRIBUTING.rst index cf3ddf6..0e01d57 100644 --- a/CONTRIBUTING.rst +++ b/CONTRIBUTING.rst @@ -95,5 +95,5 @@ http://contributor-covenant.org/version/1/1/0/. .. _Contributor Covenant: http://contributor-covenant.org .. _issue tracker: https://github.com/hylang/hy/issues .. _Fork the Repo: https://help.github.com/articles/fork-a-repo/ -.. _Start Here: http://rogerdudler.github.io/git-guide/) +.. _Start Here: http://rogerdudler.github.io/git-guide/ .. _good-first-bug: http://github.com/hylang/hy/issues?q=is%3Aissue+is%3Aopen+label%3Agood-first-bug From 108537a4e0d0c9b467ae116ffab344ec8b4ffbca Mon Sep 17 00:00:00 2001 From: Kodi Arfer Date: Tue, 11 Oct 2016 13:31:22 -0700 Subject: [PATCH 21/25] Allow &rest after &optional, like Python --- hy/compiler.py | 10 ++-------- tests/compilers/test_ast.py | 2 ++ 2 files changed, 4 insertions(+), 8 deletions(-) diff --git a/hy/compiler.py b/hy/compiler.py index 6e6e627..b6c6d27 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -503,19 +503,13 @@ class HyASTCompiler(object): for expr in exprs: if expr in ll_keywords: - if expr == "&rest" and lambda_keyword is None: - lambda_keyword = expr - elif expr == "&optional": + if expr == "&optional": if len(defaults) > 0: raise HyTypeError(expr, "There can only be &optional " "arguments or one &key argument") lambda_keyword = expr - elif expr == "&key": - lambda_keyword = expr - elif expr == "&kwonly": - lambda_keyword = expr - elif expr == "&kwargs": + elif expr in ("&rest", "&key", "&kwonly", "&kwargs"): lambda_keyword = expr else: raise HyTypeError(expr, diff --git a/tests/compilers/test_ast.py b/tests/compilers/test_ast.py index 4898c40..31c90bd 100644 --- a/tests/compilers/test_ast.py +++ b/tests/compilers/test_ast.py @@ -397,6 +397,7 @@ def test_lambda_list_keywords_rest(): """ Ensure we can compile functions with lambda list keywords.""" can_compile("(fn (x &rest xs) (print xs))") cant_compile("(fn (x &rest xs &rest ys) (print xs))") + can_compile("(fn (&optional a &rest xs) (print xs))") def test_lambda_list_keywords_key(): @@ -410,6 +411,7 @@ def test_lambda_list_keywords_kwargs(): """ Ensure we can compile functions with &kwargs.""" can_compile("(fn (x &kwargs kw) (list x kw))") cant_compile("(fn (x &kwargs xs &kwargs ys) (list x xs ys))") + can_compile("(fn (&optional x &kwargs kw) (list x kw))") def test_lambda_list_keywords_kwonly(): From 4b0296d257be392068ed1ffbab037857fa8bbc38 Mon Sep 17 00:00:00 2001 From: Kodi Arfer Date: Tue, 11 Oct 2016 14:04:50 -0700 Subject: [PATCH 22/25] Don't check the type of arguments to `inc`, `odd?`, etc. This allows them to be used with numeric types that aren't built in, such as NumPy arrays. Because Python uses duck typing, there's generally no way to know in advance whether a given value will accept a given operator. Of course, things like `(inc "hello")` will still raise a `TypeError`, because so does `(+ "hello" 1)`. --- hy/core/language.hy | 11 ------- tests/native_tests/core.hy | 59 +++++++++++--------------------------- 2 files changed, 16 insertions(+), 54 deletions(-) diff --git a/hy/core/language.hy b/hy/core/language.hy index 07bc202..b7fce7e 100644 --- a/hy/core/language.hy +++ b/hy/core/language.hy @@ -39,10 +39,6 @@ (import [hy.lex [LexException PrematureEndOfInput tokenize]]) (import [hy.compiler [HyASTCompiler]]) -(defn _numeric-check [x] - (if (not (numeric? x)) - (raise (TypeError (.format "{0!r} is not a number" x))))) - (defn butlast [coll] "Returns coll except of last element." (drop-last 1 coll)) @@ -66,7 +62,6 @@ (defn dec [n] "Decrement n by 1" - (_numeric-check n) (- n 1)) (defn disassemble [tree &optional [codegen false]] @@ -170,7 +165,6 @@ (defn even? [n] "Return true if n is an even number" - (_numeric-check n) (= (% n 2) 0)) (defn every? [pred coll] @@ -239,7 +233,6 @@ (defn inc [n] "Increment n by 1" - (_numeric-check n) (+ n 1)) (defn instance? [klass x] @@ -325,7 +318,6 @@ (defn neg? [n] "Return true if n is < 0" - (_numeric-check n) (< n 0)) (defn none? [x] @@ -347,7 +339,6 @@ (defn odd? [n] "Return true if n is an odd number" - (_numeric-check n) (= (% n 2) 1)) (def -sentinel (object)) @@ -364,7 +355,6 @@ (defn pos? [n] "Return true if n is > 0" - (_numeric_check n) (> n 0)) (defn rest [coll] @@ -415,7 +405,6 @@ (defn zero? [n] "Return true if n is 0" - (_numeric_check n) (= n 0)) (defn read [&optional [from-file sys.stdin] diff --git a/tests/native_tests/core.hy b/tests/native_tests/core.hy index 9e3ad52..ead3960 100644 --- a/tests/native_tests/core.hy +++ b/tests/native_tests/core.hy @@ -33,6 +33,11 @@ (defn assert-nil [x] (assert (is x nil))) +(defn assert-requires-num [f] + (for [x ["foo" [] None]] + (assert-true (try (do (f x) False) + (except [e [TypeError]] True))))) + (defn test-coll? [] "NATIVE: testing coll?" (assert-true (coll? [1 2 3])) @@ -66,12 +71,7 @@ (assert-equal 0 (dec 1)) (assert-equal -1 (dec 0)) (assert-equal 0 (dec (dec 2))) - (try (do (dec "foo") (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (dec []) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (dec None) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e)))))) + (assert-requires-num dec)) (defn test-setv [] "NATIVE: testing setv mutation" @@ -173,12 +173,7 @@ (assert-true (even? -2)) (assert-false (even? 1)) (assert-true (even? 0)) - (try (even? "foo") - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (even? []) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (even? None) - (except [e [TypeError]] (assert (in "not a number" (str e)))))) + (assert-requires-num even?)) (defn test-every? [] "NATIVE: testing the every? function" @@ -263,12 +258,11 @@ "NATIVE: testing the inc function" (assert-equal 3 (inc 2)) (assert-equal 0 (inc -1)) - (try (do (inc "foo") (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (inc []) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (inc None) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e)))))) + (assert-requires-num inc) + + (defclass X [object] + [__add__ (fn [self other] (.format "__add__ got {}" other))]) + (assert-equal (inc (X)) "__add__ got 1")) (defn test-instance [] "NATIVE: testing instance? function" @@ -394,24 +388,13 @@ (assert-true (neg? -2)) (assert-false (neg? 1)) (assert-false (neg? 0)) - (try (do (neg? "foo") (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (neg? []) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (neg? None) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e)))))) + (assert-requires-num neg?)) (defn test-zero [] "NATIVE: testing the zero? function" (assert-false (zero? -2)) (assert-false (zero? 1)) - (assert-true (zero? 0)) - (try (do (zero? "foo") (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (zero? []) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (zero? None) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e)))))) + (assert-true (zero? 0))) (defn test-none [] "NATIVE: testing for `is None`" @@ -463,12 +446,7 @@ (assert-true (odd? -3)) (assert-true (odd? 1)) (assert-false (odd? 0)) - (try (do (odd? "foo") (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (odd? []) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (odd? None) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e)))))) + (assert-requires-num odd?)) (defn test-partition [] "NATIVE: testing the partition function" @@ -500,12 +478,7 @@ (assert-true (pos? 2)) (assert-false (pos? -1)) (assert-false (pos? 0)) - (try (do (pos? "foo") (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (pos? []) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e))))) - (try (do (pos? None) (assert False)) - (except [e [TypeError]] (assert (in "not a number" (str e)))))) + (assert-requires-num pos?)) (defn test-remove [] "NATIVE: testing the remove function" From cd1d12565db3a645088f5a6f6517208c18c5b086 Mon Sep 17 00:00:00 2001 From: Kodi Arfer Date: Tue, 11 Oct 2016 15:48:40 -0700 Subject: [PATCH 23/25] Assertion-testing style edit --- tests/native_tests/core.hy | 5 +++-- 1 file changed, 3 insertions(+), 2 deletions(-) diff --git a/tests/native_tests/core.hy b/tests/native_tests/core.hy index ead3960..ee74b6f 100644 --- a/tests/native_tests/core.hy +++ b/tests/native_tests/core.hy @@ -35,8 +35,9 @@ (defn assert-requires-num [f] (for [x ["foo" [] None]] - (assert-true (try (do (f x) False) - (except [e [TypeError]] True))))) + (try (f x) + (except [TypeError] True) + (else (assert False))))) (defn test-coll? [] "NATIVE: testing coll?" From 0eb407676828fd6a24e9aafc5b10988026548954 Mon Sep 17 00:00:00 2001 From: Kodi Arfer Date: Tue, 11 Oct 2016 20:14:14 -0700 Subject: [PATCH 24/25] Require Python 3 for neg? and pos? type tests Python 2 allows 2 < "hello" and the like. --- tests/native_tests/core.hy | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/tests/native_tests/core.hy b/tests/native_tests/core.hy index ee74b6f..0bf1d4f 100644 --- a/tests/native_tests/core.hy +++ b/tests/native_tests/core.hy @@ -19,6 +19,8 @@ ;; FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER ;; DEALINGS IN THE SOFTWARE. +(import [hy._compat [PY3]]) + ;;;; some simple helpers (defn assert-true [x] @@ -389,7 +391,8 @@ (assert-true (neg? -2)) (assert-false (neg? 1)) (assert-false (neg? 0)) - (assert-requires-num neg?)) + (when PY3 + (assert-requires-num neg?))) (defn test-zero [] "NATIVE: testing the zero? function" @@ -479,7 +482,8 @@ (assert-true (pos? 2)) (assert-false (pos? -1)) (assert-false (pos? 0)) - (assert-requires-num pos?)) + (when PY3 + (assert-requires-num pos?))) (defn test-remove [] "NATIVE: testing the remove function" From 34304c41c7063bc3c9f25a4d150a0d9ead1eb2f2 Mon Sep 17 00:00:00 2001 From: Ryan Gonzalez Date: Wed, 12 Oct 2016 17:11:46 -0500 Subject: [PATCH 25/25] Test Python 3.6 apart from nightly (#1133) * Test Python 3.6 apart from nightly (ref. #1113) change to `3.6-dev` --- .travis.yml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/.travis.yml b/.travis.yml index 8ab3a44..bd99240 100644 --- a/.travis.yml +++ b/.travis.yml @@ -4,7 +4,7 @@ matrix: include: - python: 3.5 env: TOXENV=py35 - - python: nightly + - python: 3.6-dev env: TOXENV=py36 env: - TOXENV=py27