From b92049d11982bbd82784c959cb0e462bd9451cb6 Mon Sep 17 00:00:00 2001 From: Gergely Nagy Date: Tue, 11 Aug 2015 14:03:09 +0200 Subject: [PATCH] Allow defclass to have properties/method with built-in names To allow classes to have methods that match built-in names, yet, still disallow them outside of defclass, keep an internal state whether builtins are allowed in the current context. By default, this is false. But defclass will set it to True when it compiles its body, and set it back to the previous value when it's done with that. We need to set back to the previous value to allow nested defclasses to work properly. This closes #783. Signed-off-by: Gergely Nagy --- hy/compiler.py | 8 +++++++- tests/compilers/test_ast.py | 10 +++++++++- 2 files changed, 16 insertions(+), 2 deletions(-) diff --git a/hy/compiler.py b/hy/compiler.py index d71c5df..3be5f21 100644 --- a/hy/compiler.py +++ b/hy/compiler.py @@ -370,6 +370,7 @@ def checkargs(exact=None, min=None, max=None, even=None, multiple=None): class HyASTCompiler(object): def __init__(self, module_name): + self.allow_builtins = False self.anon_fn_count = 0 self.anon_var_count = 0 self.imports = defaultdict(set) @@ -2005,7 +2006,8 @@ class HyASTCompiler(object): start_line, start_column): str_name = "%s" % name - if _is_hy_builtin(str_name, self.module_name): + if _is_hy_builtin(str_name, self.module_name) and \ + not self.allow_builtins: raise HyTypeError(name, "Can't assign to a builtin: `%s'" % str_name) @@ -2275,6 +2277,8 @@ class HyASTCompiler(object): docstring.start_column) body += body.expr_as_stmt() + allow_builtins = self.allow_builtins + self.allow_builtins = True if expressions and isinstance(expressions[0], HyList) \ and not isinstance(expressions[0], HyExpression): expr = expressions.pop(0) @@ -2287,6 +2291,8 @@ class HyASTCompiler(object): for expression in expressions: body += self.compile(macroexpand(expression, self.module_name)) + self.allow_builtins = allow_builtins + if not body.stmts: body += ast.Pass(lineno=expressions.start_line, col_offset=expressions.start_column) diff --git a/tests/compilers/test_ast.py b/tests/compilers/test_ast.py index 0a6c36e..8660a11 100644 --- a/tests/compilers/test_ast.py +++ b/tests/compilers/test_ast.py @@ -554,6 +554,14 @@ def test_defn(): def test_setv_builtins(): - """Ensure that assigning to a builtin fails""" + """Ensure that assigning to a builtin fails, unless in a class""" cant_compile("(setv nil 42)") cant_compile("(defn get [&rest args] 42)") + can_compile("(defclass A [] (defn get [self] 42))") + can_compile(""" + (defclass A [] + (defn get [self] 42) + (defclass B [] + (defn get [self] 42)) + (defn if [self] 0)) + """)