Merge branch 'master' into pr/185

Conflicts:
	hy/macros.py
	tests/compilers/test_compiler.py

Hoodoggy!
This commit is contained in:
Paul Tagliamonte 2013-05-17 11:17:51 -04:00
commit 9c28e0292d
9 changed files with 98 additions and 54 deletions

View File

@ -99,7 +99,7 @@ def builds(_type):
unpythonic_chars = ["-"] unpythonic_chars = ["-"]
really_ok = ["-"] 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: if _type not in really_ok:
raise TypeError("`build' needs to be *post* translated strings, " raise TypeError("`build' needs to be *post* translated strings, "
"Mr. / Mrs. Hypser. -- `%s' sucks." % (_type)) "Mr. / Mrs. Hypser. -- `%s' sucks." % (_type))
@ -422,13 +422,14 @@ class HyASTCompiler(object):
if isinstance(expr, HyLambdaListKeyword): if isinstance(expr, HyLambdaListKeyword):
if expr not in expr._valid_types: if expr not in expr._valid_types:
raise HyCompileError("{0} is not a valid " raise HyTypeError(expr, "{0} is not a valid "
"lambda-keyword.".format(repr(expr))) "lambda-keyword.".format(repr(expr)))
if expr == "&rest" and lambda_keyword is None: if expr == "&rest" and lambda_keyword is None:
lambda_keyword = expr lambda_keyword = expr
elif expr == "&optional": elif expr == "&optional":
if len(defaults) > 0: if len(defaults) > 0:
raise HyCompileError("There can only be &optional " raise HyTypeError(expr,
"There can only be &optional "
"arguments or one &key argument") "arguments or one &key argument")
lambda_keyword = expr lambda_keyword = expr
elif expr == "&key": elif expr == "&key":
@ -436,7 +437,8 @@ class HyASTCompiler(object):
elif expr == "&kwargs": elif expr == "&kwargs":
lambda_keyword = expr lambda_keyword = expr
else: else:
raise HyCompileError("{0} is in an invalid " raise HyTypeError(expr,
"{0} is in an invalid "
"position.".format(repr(expr))) "position.".format(repr(expr)))
# we don't actually care about this token, so we set # we don't actually care about this token, so we set
# our state and continue to the next token... # our state and continue to the next token...
@ -446,27 +448,32 @@ class HyASTCompiler(object):
args.append(expr) args.append(expr)
elif lambda_keyword == "&rest": elif lambda_keyword == "&rest":
if varargs: if varargs:
raise HyCompileError("There can only be one " raise HyTypeError(expr,
"There can only be one "
"&rest argument") "&rest argument")
varargs = str(expr) varargs = str(expr)
elif lambda_keyword == "&key": elif lambda_keyword == "&key":
if type(expr) != HyDict: if type(expr) != HyDict:
raise TypeError("There can only be one &key " raise HyTypeError(expr,
"There can only be one &key "
"argument") "argument")
else: else:
if len(defaults) > 0: if len(defaults) > 0:
raise HyCompileError("There can only be &optional " raise HyTypeError(expr,
"There can only be &optional "
"arguments or one &key argument") "arguments or one &key argument")
# As you can see, Python has a funny way of # As you can see, Python has a funny way of
# defining keyword arguments. # defining keyword arguments.
for k, v in expr.items(): it = iter(expr)
for k, v in zip(it, it):
args.append(k) args.append(k)
ret += self.compile(v) ret += self.compile(v)
defaults.append(ret.force_expr) defaults.append(ret.force_expr)
elif lambda_keyword == "&optional": elif lambda_keyword == "&optional":
if isinstance(expr, HyList): if isinstance(expr, HyList):
if not len(expr) == 2: if not len(expr) == 2:
raise TypeError("optional args should be bare names " raise HyTypeError(expr,
"optional args should be bare names "
"or 2-item lists") "or 2-item lists")
k, v = expr k, v = expr
else: else:
@ -477,7 +484,8 @@ class HyASTCompiler(object):
defaults.append(ret.force_expr) defaults.append(ret.force_expr)
elif lambda_keyword == "&kwargs": elif lambda_keyword == "&kwargs":
if kwargs: if kwargs:
raise HyCompileError("There can only be one " raise HyTypeError(expr,
"There can only be one "
"&kwargs argument") "&kwargs argument")
kwargs = str(expr) kwargs = str(expr)
@ -546,7 +554,7 @@ class HyASTCompiler(object):
name = form.__class__.__name__ name = form.__class__.__name__
imports = set([name]) imports = set([name])
if isinstance(form, HyList): if isinstance(form, (HyList, HyDict)):
if not form: if not form:
contents = HyList() contents = HyList()
else: else:
@ -764,7 +772,8 @@ class HyASTCompiler(object):
@builds("except") @builds("except")
@builds("catch") @builds("catch")
def magic_internal_form(self, expr): 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): def _compile_catch_expression(self, expr, var):
catch = expr.pop(0) # catch catch = expr.pop(0) # catch
@ -796,6 +805,11 @@ class HyASTCompiler(object):
# let's pop variable and use it as name # let's pop variable and use it as name
if len(exceptions) == 2: if len(exceptions) == 2:
name = exceptions.pop(0) 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: if sys.version_info[0] >= 3:
# Python3 features a change where the Exception handler # Python3 features a change where the Exception handler
# moved the name from a Name() to a pure Python String type. # moved the name from a Name() to a pure Python String type.
@ -1694,7 +1708,7 @@ class HyASTCompiler(object):
@builds(HyDict) @builds(HyDict)
def compile_dict(self, m): 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, ret += ast.Dict(lineno=m.start_line,
col_offset=m.start_column, col_offset=m.start_column,

View File

@ -228,9 +228,7 @@ class Dict(ListeyThing):
def exit(self): def exit(self):
self.commit() self.commit()
it = iter(self.nodes) self.result = HyDict(self.nodes)
result = dict(zip(it, it))
self.result = HyDict(result)
end_char = "}" end_char = "}"

View File

@ -20,7 +20,6 @@
from hy.models.expression import HyExpression from hy.models.expression import HyExpression
from hy.models.string import HyString from hy.models.string import HyString
from hy.models.dict import HyDict
from hy.models.list import HyList from hy.models.list import HyList
from collections import defaultdict from collections import defaultdict
@ -65,15 +64,8 @@ def process(tree, module_name):
ntree.replace(tree) ntree.replace(tree)
return ntree return ntree
if isinstance(tree, HyDict):
obj = HyDict(dict((process(x, module_name),
process(tree[x], module_name))
for x in tree))
obj.replace(tree)
return obj
if isinstance(tree, HyList): if isinstance(tree, HyList):
obj = HyList([process(x, module_name) for x in tree]) # NOQA obj = tree.__class__([process(x, module_name) for x in tree]) # NOQA
# flake8 thinks we're redefining from 52. # flake8 thinks we're redefining from 52.
obj.replace(tree) obj.replace(tree)
return obj return obj

View File

@ -18,17 +18,13 @@
# FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER # FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER
# DEALINGS IN THE SOFTWARE. # 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): def __repr__(self):
for x in self: return "{%s}" % (" ".join([repr(x) for x in self]))
self[x].replace(other)
x.replace(other)
HyObject.replace(self, other)

View File

@ -22,7 +22,7 @@
from __future__ import unicode_literals from __future__ import unicode_literals
from hy import HyString from hy import HyString
from hy.compiler import hy_compile, HyCompileError from hy.compiler import hy_compile, HyCompileError, HyTypeError
from hy.lex import tokenize from hy.lex import tokenize
import ast import ast
@ -47,8 +47,11 @@ def cant_compile(expr):
try: try:
hy_compile(expr, "__main__") hy_compile(expr, "__main__")
assert False assert False
except HyCompileError: except HyCompileError as e:
pass # 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(): def test_ast_bad_type():

View File

@ -27,11 +27,21 @@ if sys.version_info[0] <= 2 and sys.version_info[1] <= 6:
else: else:
import unittest import unittest
from hy import compiler
from hy.models.expression import HyExpression from hy.models.expression import HyExpression
from hy.models.list import HyList from hy.models.list import HyList
from hy.models.symbol import HySymbol 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): class HyASTCompilerTest(unittest.TestCase):
@ -46,7 +56,7 @@ class HyASTCompilerTest(unittest.TestCase):
return h return h
def setUp(self): def setUp(self):
self.c = HyASTCompiler('') self.c = compiler.HyASTCompiler('test')
def test_fn_compiler_empty_function(self): def test_fn_compiler_empty_function(self):
ret = self.c.compile_function_def( ret = self.c.compile_function_def(

View File

@ -185,15 +185,18 @@ def test_lex_line_counting_multi_inner():
def test_dicts(): def test_dicts():
""" Ensure that we can tokenize a dict. """ """ Ensure that we can tokenize a dict. """
objs = tokenize("{foo bar bar baz}") objs = tokenize("{foo bar bar baz}")
assert objs == [HyDict({ assert objs == [HyDict(["foo", "bar", "bar", "baz"])]
"foo": "bar",
"bar": "baz"
})]
objs = tokenize("(bar {foo bar bar baz})") objs = tokenize("(bar {foo bar bar baz})")
assert objs == [HyExpression([HySymbol("bar"), assert objs == [HyExpression([HySymbol("bar"),
HyDict({"foo": "bar", HyDict(["foo", "bar",
"bar": "baz"})])] "bar", "baz"])])]
objs = tokenize("{(foo bar) (baz quux)}")
assert objs == [HyDict([
HyExpression([HySymbol("foo"), HySymbol("bar")]),
HyExpression([HySymbol("baz"), HySymbol("quux")])
])]
def test_nospace(): def test_nospace():

View File

@ -15,6 +15,12 @@
(assert (= [1 2 3 4] (+ [1 2] [3 4])))) (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 [] (defn test-setv-get []
"NATIVE: test setv works on a get expression" "NATIVE: test setv works on a get expression"
(setv foo [0 1 2]) (setv foo [0 1 2])

View File

@ -1,3 +1,6 @@
(import hy)
(defn test-quote [] (defn test-quote []
"NATIVE: test for quoting functionality" "NATIVE: test for quoting functionality"
(setf q (quote (a b c))) (setf q (quote (a b c)))
@ -21,6 +24,25 @@
(assert (= (cdr q1) (quote (a b c))))) (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 [] (defn test-quasiquote []
"NATIVE: test that quasiquote and quote are equivalent for simple cases" "NATIVE: test that quasiquote and quote are equivalent for simple cases"
(setf q (quote (a b c))) (setf q (quote (a b c)))