Merge branch 'jd/catch' of git://github.com/jd/hy into pr-73

This commit is contained in:
Paul R. Tagliamonte 2013-04-07 12:23:27 -04:00
commit 70fd991ef1
3 changed files with 124 additions and 18 deletions

View File

@ -1,4 +1,7 @@
# -*- encoding: utf-8 -*-
#
# Copyright (c) 2013 Paul Tagliamonte <paultag@debian.org> # Copyright (c) 2013 Paul Tagliamonte <paultag@debian.org>
# Copyright (c) 2013 Julien Danjou <julien@danjou.info>
# #
# Permission is hereby granted, free of charge, to any person obtaining a # Permission is hereby granted, free of charge, to any person obtaining a
# copy of this software and associated documentation files (the "Software"), # copy of this software and associated documentation files (the "Software"),
@ -191,12 +194,30 @@ class HyASTCompiler(object):
orelse=[]) orelse=[])
@builds("catch") @builds("catch")
@checkargs(min=2)
def compile_catch_expression(self, expr): def compile_catch_expression(self, expr):
expr.pop(0) # catch expr.pop(0) # catch
_type = self.compile(expr.pop(0))
name = expr.pop(0)
try:
exceptions = expr.pop(0)
except IndexError:
exceptions = []
# exceptions catch should be either:
# [[list of exceptions]]
# or
# [variable [list of exceptions]]
# or
# [variable exception]
# or
# [exception]
# or
# []
if len(exceptions) > 2:
raise TypeError("`catch' exceptions list is too long")
# [variable [list of exceptions]]
# let's pop variable and use it as name
if len(exceptions) == 2:
name = exceptions.pop(0)
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.
@ -207,13 +228,43 @@ class HyASTCompiler(object):
else: else:
# Python2 requires an ast.Name, set to ctx Store. # Python2 requires an ast.Name, set to ctx Store.
name = self._storeize(self.compile(name)) name = self._storeize(self.compile(name))
else:
name = None
try:
exceptions_list = exceptions.pop(0)
except IndexError:
exceptions_list = []
if isinstance(exceptions_list, list):
if len(exceptions_list):
# [FooBar BarFoo] → catch Foobar and BarFoo exceptions
_type = ast.Tuple(elts=[self.compile(x)
for x in exceptions_list],
lineno=expr.start_line,
col_offset=expr.start_column,
ctx=ast.Load())
else:
# [] → all exceptions catched
_type = None
elif isinstance(exceptions_list, HySymbol):
_type = self.compile(exceptions_list)
else:
raise TypeError("`catch' needs a valid exception list to catch")
if len(expr) == 0:
# No body
body = [ast.Pass(lineno=expr.start_line,
col_offset=expr.start_column)]
else:
body = self._code_branch([self.compile(x) for x in expr])
return ast.ExceptHandler( return ast.ExceptHandler(
lineno=expr.start_line, lineno=expr.start_line,
col_offset=expr.start_column, col_offset=expr.start_column,
type=_type, type=_type,
name=name, name=name,
body=self._code_branch([self.compile(x) for x in expr])) body=body)
def _code_branch(self, branch): def _code_branch(self, branch):
if isinstance(branch, list): if isinstance(branch, list):

View File

@ -116,13 +116,20 @@ def test_ast_bad_try():
def test_ast_good_catch(): def test_ast_good_catch():
"Make sure AST can compile valid catch" "Make sure AST can compile valid catch"
hy_compile(tokenize("(catch 1 2)")) 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]])"))
def test_ast_bad_catch(): def test_ast_bad_catch():
"Make sure AST can't compile invalid catch" "Make sure AST can't compile invalid catch"
cant_compile("(catch)")
cant_compile("(catch 1)") cant_compile("(catch 1)")
cant_compile("(catch [1 3])")
cant_compile("(catch [x [FooBar] BarBar]])")
def test_ast_good_assert(): def test_ast_good_assert():

View File

@ -164,8 +164,56 @@
"NATIVE: test Exceptions" "NATIVE: test Exceptions"
(try (try
(throw (KeyError)) (throw (KeyError))
(catch IOError e (assert (= 2 1))) (catch [[IOError]] (assert false))
(catch KeyError e (+ 1 1) (assert (= 1 1))))) (catch [e [KeyError]] (assert e)))
(try
(get [1] 3)
(catch [IndexError] (assert true))
(catch [IndexError] (pass)))
(try
(print foobar42ofthebaz)
(catch [IndexError] (assert false))
(catch [NameError] (pass)))
(try
(get [1] 3)
(catch [e IndexError] (assert (isinstance e IndexError))))
(try
(get [1] 3)
(catch [e [IndexError NameError]] (assert (isinstance e IndexError))))
(try
(print foobar42ofthebaz)
(catch [e [IndexError NameError]] (assert (isinstance e NameError))))
(try
(print foobar42)
(catch [[IndexError NameError]] (pass)))
(try
(get [1] 3)
(catch [[IndexError NameError]] (pass)))
(try
(print foobar42ofthebaz)
(catch))
(try
(print foobar42ofthebaz)
(catch []))
(try
(print foobar42ofthebaz)
(catch [] (pass)))
(try
(print foobar42ofthebaz)
(catch []
(setv foobar42ofthebaz 42)
(assert (= foobar42ofthebaz 42)))))
(defn test-earmuffs [] (defn test-earmuffs []
"NATIVE: Test earmuffs" "NATIVE: Test earmuffs"
@ -327,7 +375,7 @@
6)) 6))
(try (try
(assert (= x 42)) ; This ain't true (assert (= x 42)) ; This ain't true
(catch NameError e (assert e))) (catch [e [NameError]] (assert e)))
(assert (= y 123))) (assert (= y 123)))