Skip to content

Commit

Permalink
Merge pull request #1588 from Kodiologist/no-amp-key
Browse files Browse the repository at this point in the history
Remove &key
  • Loading branch information
vodik authored Apr 27, 2018
2 parents fbeff8d + a605936 commit 032247e
Show file tree
Hide file tree
Showing 10 changed files with 7 additions and 109 deletions.
1 change: 1 addition & 0 deletions NEWS.rst
Original file line number Diff line number Diff line change
Expand Up @@ -8,6 +8,7 @@ Removals
* Dotted lists, `HyCons`, `cons`, `cons?`, and `list*` have been removed.
These were redundant with Python's built-in data structures and Hy's most
common model types (`HyExpression`, `HyList`, etc.).
* `&key` is no longer special in lambda lists. Use `&optional` instead.

Other Breaking Changes
------------------------------
Expand Down
22 changes: 0 additions & 22 deletions docs/language/api.rst
Original file line number Diff line number Diff line change
Expand Up @@ -493,28 +493,6 @@ Parameters may have the following keywords in front of them:
=> (total-value 100 1)
101.0

&key
Parameter is a dict of keyword arguments. The keys of the dict
specify the parameter names and the values give the default values
of the parameters.

.. code-block:: clj
=> (defn key-parameters [&key {"a" 1 "b" 2}]
... (print "a is" a "and b is" b))
=> (key-parameters :a 1 :b 2)
a is 1 and b is 2
=> (key-parameters :b 1 :a 2)
a is 2 and b is 1
The following declarations are equivalent:

.. code-block:: clj
(defn key-parameters [&key {"a" 1 "b" 2}])
(defn key-parameters [&optional [a 1] [b 2]])
&kwargs
Parameter will contain 0 or more keyword arguments.

Expand Down
11 changes: 0 additions & 11 deletions docs/tutorial.rst
Original file line number Diff line number Diff line change
Expand Up @@ -443,17 +443,6 @@ similar to `*args` and `**kwargs` in Python::
=> (optional-arg #* args #** kwargs)
[1, 2, 4, 3]

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 parameter lists. In Python::

def some_func(foo, bar, *args, **kwargs):
Expand Down
31 changes: 2 additions & 29 deletions hy/compiler.py
Original file line number Diff line number Diff line change
Expand Up @@ -529,7 +529,7 @@ def _compile_branch(self, exprs):

def _parse_lambda_list(self, exprs):
""" Return FunctionDef parameter values from lambda list."""
ll_keywords = ("&rest", "&optional", "&key", "&kwonly", "&kwargs")
ll_keywords = ("&rest", "&optional", "&kwonly", "&kwargs")
ret = Result()
args = []
defaults = []
Expand All @@ -542,13 +542,7 @@ def _parse_lambda_list(self, exprs):
for expr in exprs:

if expr in ll_keywords:
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 in ("&rest", "&key", "&kwonly", "&kwargs"):
if expr in ("&optional", "&rest", "&kwonly", "&kwargs"):
lambda_keyword = expr
else:
raise HyTypeError(expr,
Expand All @@ -566,27 +560,6 @@ def _parse_lambda_list(self, exprs):
"There can only be one "
"&rest argument")
varargs = expr
elif lambda_keyword == "&key":
if type(expr) != HyDict:
raise HyTypeError(expr,
"There can only be one &key "
"argument")
else:
if len(defaults) > 0:
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.
it = iter(expr)
for k, v in zip(it, it):
if not isinstance(k, HyString):
raise HyTypeError(expr,
"Only strings can be used "
"as parameter names")
args.append(k)
ret += self.compile(v)
defaults.append(ret.force_expr)
elif lambda_keyword == "&optional":
if isinstance(expr, HyList):
if not len(expr) == 2:
Expand Down
11 changes: 2 additions & 9 deletions hy/contrib/walk.hy
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,7 @@ splits a fn argument list into sections based on &-headers.
returns an OrderedDict mapping headers to sublists.
Arguments without a header are under None.
"
(setv headers '[&optional &key &rest &kwonly &kwargs]
(setv headers '[&optional &rest &kwonly &kwargs]
sections (OrderedDict [(, None [])])
header None)
(for [arg form]
Expand Down Expand Up @@ -177,14 +177,7 @@ Arguments without a header are under None.
~(self.expand-symbols (second pair))])]
[True
(.add protected pair)
(.append argslist pair)]))]
[(= header '&key)
(setv &key-dict '{})
(for [[k v] (-> section first partition)]
(.add protected k)
(.append &key-dict k)
(.append &key-dict (self.expand-symbols v)))
(.append argslist &key-dict)]))
(.append argslist pair)]))]))
(, protected argslist))

(defn handle-fn [self]
Expand Down
2 changes: 1 addition & 1 deletion hy/core/bootstrap.hy
Original file line number Diff line number Diff line change
Expand Up @@ -18,7 +18,7 @@
(% "received a `%s' instead of a symbol for macro name"
(. (type name)
__name__)))))
(for* [kw '[&kwonly &kwargs &key]]
(for* [kw '[&kwonly &kwargs]]
(if* (in kw lambda-list)
(raise (hy.errors.HyTypeError macro-name
(% "macros cannot use %s"
Expand Down
9 changes: 0 additions & 9 deletions tests/compilers/test_ast.py
Original file line number Diff line number Diff line change
Expand Up @@ -396,8 +396,6 @@ def test_ast_non_decoratable():

def test_ast_lambda_lists():
"""Ensure the compiler chokes on invalid lambda-lists"""
cant_compile('(fn [&key {"a" b} &key {"foo" bar}] [a foo])')
cant_compile('(fn [&optional a &key {"foo" bar}] [a foo])')
cant_compile('(fn [&optional [a b c]] a)')
cant_compile('(fn [&optional [1 2]] (list 1 2))')

Expand Down Expand Up @@ -427,13 +425,6 @@ def test_lambda_list_keywords_rest():
can_compile("(fn (&optional a &rest xs) (print xs))")


def test_lambda_list_keywords_key():
""" Ensure we can compile functions with &key."""
can_compile("(fn (x &key {foo True}) (list x foo))")
cant_compile("(fn (x &key {bar \"baz\"} &key {foo 42}) (list x bar foo))")
cant_compile("(fn (x &key {1 2 3 4}) (list x))")


def test_lambda_list_keywords_kwargs():
""" Ensure we can compile functions with &kwargs."""
can_compile("(fn (x &kwargs kw) (list x kw))")
Expand Down
13 changes: 0 additions & 13 deletions tests/native_tests/contrib/walk.hy
Original file line number Diff line number Diff line change
Expand Up @@ -295,19 +295,6 @@
(assert (= (foo 10 20 30)
(, 10 20 30)))))

(defn test-let-key []
(let [a 1
b 6
d 2]
(defn foo [&key {a a b None c d}]
(, a b c))
(assert (= (foo)
(, 1 None 2)))
(assert (= (foo 10 20 30)
(, 10 20 30)))
(assert (= (, a b d)
(, 1 6 2)))))

(defn test-let-closure []
(let [count 0]
(defn +count [&optional [x 1]]
Expand Down
8 changes: 0 additions & 8 deletions tests/native_tests/language.hy
Original file line number Diff line number Diff line change
Expand Up @@ -1374,14 +1374,6 @@
(assert (= (foo 10 20 30) [10 (, 20 30) {}])))


(defn test-key-arguments []
"NATIVE: test &key function arguments"
(defn foo [&key {"a" None "b" 1}] [a b])
(assert (= (foo) [None 1]))
(assert (= (foo :a 2) [2 1]))
(assert (= (foo :b 42) [None 42])))


(defn test-optional-arguments []
"NATIVE: test &optional function arguments"
(defn foo [a b &optional c [d 42]] [a b c d])
Expand Down
8 changes: 1 addition & 7 deletions tests/native_tests/native_macros.hy
Original file line number Diff line number Diff line change
Expand Up @@ -61,7 +61,7 @@
(foo x y))

(defn test-macro-kw []
"NATIVE: test that an error is raised when &kwonly, &kwargs, or &key is used in a macro"
"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]
Expand All @@ -72,12 +72,6 @@
(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 []
Expand Down

0 comments on commit 032247e

Please sign in to comment.