Use a fixed compiler in HyREPL

These changes make `HyREPL` use a single `HyASTCompiler` instance, instead of
creating one every time a valid source string is processed.

This change avoids the unnecessary re-initiation of the standard library
`import` and `require` steps that currently occur within the module tracked by a
`HyREPL` instance.

Also, one can now pass an existing compiler instance to `hy_repl` and
`hy_compiler`.

Closes hylang/hy#1698.
This commit is contained in:
Brandon T. Willard 2018-11-10 22:43:39 -06:00
parent fb0220bd52
commit 15c68455ec
2 changed files with 53 additions and 29 deletions

View File

@ -20,7 +20,8 @@ import astor.code_gen
import hy import hy
from hy.lex import hy_parse, mangle from hy.lex import hy_parse, mangle
from hy.lex.exceptions import LexException, PrematureEndOfInput from hy.lex.exceptions import LexException, PrematureEndOfInput
from hy.compiler import HyTypeError, hy_compile, hy_eval from hy.compiler import HyASTCompiler, hy_compile, hy_eval
from hy.errors import HyTypeError
from hy.importer import runhy from hy.importer import runhy
from hy.completer import completion, Completer from hy.completer import completion, Completer
from hy.macros import macro, require from hy.macros import macro, require
@ -69,6 +70,8 @@ class HyREPL(code.InteractiveConsole, object):
# Load cmdline-specific macros. # Load cmdline-specific macros.
require('hy.cmdline', module_name, assignments='ALL') require('hy.cmdline', module_name, assignments='ALL')
self.hy_compiler = HyASTCompiler(self.module)
self.spy = spy self.spy = spy
if output_fn is None: if output_fn is None:
@ -117,7 +120,10 @@ class HyREPL(code.InteractiveConsole, object):
new_ast = ast.Module(main_ast.body + new_ast = ast.Module(main_ast.body +
[ast.Expr(expr_ast.body)]) [ast.Expr(expr_ast.body)])
print(astor.to_source(new_ast)) print(astor.to_source(new_ast))
value = hy_eval(do, self.locals, self.module, ast_callback)
value = hy_eval(do, self.locals,
ast_callback=ast_callback,
compiler=self.hy_compiler)
except HyTypeError as e: except HyTypeError as e:
if e.source is None: if e.source is None:
e.source = source e.source = source

View File

@ -1779,18 +1779,38 @@ class HyASTCompiler(object):
return ret + asty.Dict(m, keys=keyvalues[::2], values=keyvalues[1::2]) return ret + asty.Dict(m, keys=keyvalues[::2], values=keyvalues[1::2])
def hy_eval(hytree, locals=None, module=None, ast_callback=None): def get_compiler_module(module=None, compiler=None, calling_frame=False):
"""Evaluates a quoted expression and returns the value. """Get a module object from a compiler, given module object,
string name of a module, and (optionally) the calling frame; otherwise,
raise an error."""
module = getattr(compiler, 'module', None) or module
if isinstance(module, string_types):
if module.startswith('<') and module.endswith('>'):
module = types.ModuleType(module)
else:
module = importlib.import_module(ast_str(module, piecewise=True))
if calling_frame and not module:
module = calling_module(n=2)
if not inspect.ismodule(module):
raise TypeError('Invalid module type: {}'.format(type(module)))
return module
def hy_eval(hytree, locals=None, module=None, ast_callback=None,
compiler=None):
"""Evaluates a quoted expression and returns the value.
Examples Examples
-------- --------
=> (eval '(print "Hello World")) => (eval '(print "Hello World"))
"Hello World" "Hello World"
If you want to evaluate a string, use ``read-str`` to convert it to a If you want to evaluate a string, use ``read-str`` to convert it to a
form first: form first:
=> (eval (read-str "(+ 1 1)")) => (eval (read-str "(+ 1 1)"))
2 2
@ -1806,25 +1826,25 @@ def hy_eval(hytree, locals=None, module=None, ast_callback=None):
module: str or types.ModuleType, optional module: str or types.ModuleType, optional
Module, or name of the module, to which the Hy tree is assigned and Module, or name of the module, to which the Hy tree is assigned and
the global values are taken. the global values are taken.
Defaults to the calling frame's module, if any, and '__eval__' The module associated with `compiler` takes priority over this value.
otherwise. When neither `module` nor `compiler` is specified, the calling frame's
module is used.
ast_callback: callable, optional ast_callback: callable, optional
A callback that is passed the Hy compiled tree and resulting A callback that is passed the Hy compiled tree and resulting
expression object, in that order, after compilation but before expression object, in that order, after compilation but before
evaluation. evaluation.
compiler: HyASTCompiler, optional
An existing Hy compiler to use for compilation. Also serves as
the `module` value when given.
Returns Returns
------- -------
out : Result of evaluating the Hy compiled tree. out : Result of evaluating the Hy compiled tree.
""" """
if module is None:
module = calling_module()
if isinstance(module, string_types): module = get_compiler_module(module, compiler, True)
module = importlib.import_module(ast_str(module, piecewise=True))
elif not inspect.ismodule(module):
raise TypeError('Invalid module type: {}'.format(type(module)))
if locals is None: if locals is None:
frame = inspect.stack()[1][0] frame = inspect.stack()[1][0]
@ -1833,7 +1853,8 @@ def hy_eval(hytree, locals=None, module=None, ast_callback=None):
if not isinstance(locals, dict): if not isinstance(locals, dict):
raise TypeError("Locals must be a dictionary") raise TypeError("Locals must be a dictionary")
_ast, expr = hy_compile(hytree, module, get_expr=True) _ast, expr = hy_compile(hytree, module=module, get_expr=True,
compiler=compiler)
# Spoof the positions in the generated ast... # Spoof the positions in the generated ast...
for node in ast.walk(_ast): for node in ast.walk(_ast):
@ -1856,14 +1877,15 @@ def hy_eval(hytree, locals=None, module=None, ast_callback=None):
return eval(ast_compile(expr, "<eval>", "eval"), globals, locals) return eval(ast_compile(expr, "<eval>", "eval"), globals, locals)
def hy_compile(tree, module, root=ast.Module, get_expr=False): def hy_compile(tree, module=None, root=ast.Module, get_expr=False,
""" compiler=None):
Compile a Hy tree into a Python AST tree. """Compile a Hy tree into a Python AST tree.
Parameters Parameters
---------- ----------
module: str or types.ModuleType module: str or types.ModuleType, optional
Module, or name of the module, in which the Hy tree is evaluated. Module, or name of the module, in which the Hy tree is evaluated.
The module associated with `compiler` takes priority over this value.
root: ast object, optional (ast.Module) root: ast object, optional (ast.Module)
Root object for the Python AST tree. Root object for the Python AST tree.
@ -1871,26 +1893,22 @@ def hy_compile(tree, module, root=ast.Module, get_expr=False):
get_expr: bool, optional (False) get_expr: bool, optional (False)
If true, return a tuple with `(root_obj, last_expression)`. If true, return a tuple with `(root_obj, last_expression)`.
compiler: HyASTCompiler, optional
An existing Hy compiler to use for compilation. Also serves as
the `module` value when given.
Returns Returns
------- -------
out : A Python AST tree out : A Python AST tree
""" """
module = get_compiler_module(module, compiler, False)
if isinstance(module, string_types):
if module.startswith('<') and module.endswith('>'):
module = types.ModuleType(module)
else:
module = importlib.import_module(ast_str(module, piecewise=True))
if not inspect.ismodule(module):
raise TypeError('Invalid module type: {}'.format(type(module)))
tree = wrap_value(tree) tree = wrap_value(tree)
if not isinstance(tree, HyObject): if not isinstance(tree, HyObject):
raise HyCompileError("`tree` must be a HyObject or capable of " raise HyCompileError("`tree` must be a HyObject or capable of "
"being promoted to one") "being promoted to one")
compiler = HyASTCompiler(module) compiler = compiler or HyASTCompiler(module)
result = compiler.compile(tree) result = compiler.compile(tree)
expr = result.force_expr expr = result.force_expr