Merge pull request #1673 from brandonwillard/fix-docstring-with-imports

Fix AST handling of docstrings and __future__ ordering
This commit is contained in:
Kodi Arfer 2018-09-11 16:32:57 -04:00 committed by GitHub
commit a2f3a08750
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 41 additions and 1 deletions

View File

@ -22,6 +22,8 @@ Bug Fixes
attribute access. attribute access.
* Fixed crashes on Windows when calling `hy-repr` on date and time * Fixed crashes on Windows when calling `hy-repr` on date and time
objects. objects.
* Fixed errors from `from __future__ import ...` statements and missing Hy
module docstrings caused by automatic importing of Hy builtins.
0.15.0 0.15.0
============================== ==============================

View File

@ -1719,7 +1719,19 @@ def hy_compile(tree, module_name, root=ast.Module, get_expr=False):
if not get_expr: if not get_expr:
result += result.expr_as_stmt() result += result.expr_as_stmt()
body = compiler.imports_as_stmts(tree) + result.stmts body = []
# Pull out a single docstring and prepend to the resulting body.
if (len(result.stmts) > 0 and
issubclass(root, ast.Module) and
isinstance(result.stmts[0], ast.Expr) and
isinstance(result.stmts[0].value, ast.Str)):
body += [result.stmts.pop(0)]
body += sorted(compiler.imports_as_stmts(tree) + result.stmts,
key=lambda a: not (isinstance(a, ast.ImportFrom) and
a.module == '__future__'))
ret = root(body=body) ret = root(body=body)

View File

@ -664,3 +664,25 @@ def test_ast_bad_yield_from():
def test_eval_generator_with_return(): def test_eval_generator_with_return():
"""Ensure generators with a return statement works.""" """Ensure generators with a return statement works."""
can_eval("(fn [] (yield 1) (yield 2) (return))") can_eval("(fn [] (yield 1) (yield 2) (return))")
def test_futures_imports():
"""Make sure __future__ imports go first, especially when builtins are
automatically added (e.g. via use of a builtin name like `name`)."""
hy_ast = can_compile((
'(import [__future__ [print_function]])\n'
'(import sys)\n'
'(setv name [1 2])'
'(print (first name))'))
assert hy_ast.body[0].module == '__future__'
assert hy_ast.body[1].module == 'hy.core.language'
hy_ast = can_compile((
'(import sys)\n'
'(import [__future__ [print_function]])\n'
'(setv name [1 2])'
'(print (first name))'))
assert hy_ast.body[0].module == '__future__'
assert hy_ast.body[1].module == 'hy.core.language'

View File

@ -4,6 +4,7 @@
;; This Hy module is intended to concisely demonstrate all of ;; This Hy module is intended to concisely demonstrate all of
;; Python's major syntactic features for the purpose of testing hy2py. ;; Python's major syntactic features for the purpose of testing hy2py.
"This is a module docstring."
(setv mystring (* "foo" 3)) (setv mystring (* "foo" 3))

View File

@ -24,6 +24,9 @@ def test_hy2py_import(tmpdir):
def assert_stuff(m): def assert_stuff(m):
# This makes sure that automatically imported builtins go after docstrings.
assert m.__doc__ == u'This is a module docstring.'
assert m.mystring == "foofoofoo" assert m.mystring == "foofoofoo"
assert m.long_string == u"This is a very long string literal, which would surely exceed any limitations on how long a line or a string literal can be. The string literal alone exceeds 256 characters. It also has a character outside the Basic Multilingual Plane: 😂. Here's a double quote: \". Here are some escaped newlines:\n\n\nHere is a literal newline:\nCall me Ishmael. Some years ago—never mind how long precisely—having little or no money in my purse, and nothing particular to interest me on shore, I thought I would sail about a little and see the watery part of the world. It is a way I have of driving off the spleen and regulating the circulation. Whenever I find myself growing grim about the mouth; whenever it is a damp, drizzly November in my soul; whenever I find myself involuntarily pausing before coffin warehouses, and bringing up the rear of every funeral I meet; and especially whenever my hypos get such an upper hand of me, that it requires a strong moral principle to prevent me from deliberately stepping into the street, and methodically knocking peoples hats off—then, I account it high time to get to sea as soon as I can. This is my substitute for pistol and ball. With a philosophical flourish Cato throws himself upon his sword; I quietly take to the ship. There is nothing surprising in this. If they but knew it, almost all men in their degree, some time or other, cherish very nearly the same feelings towards the ocean with me." assert m.long_string == u"This is a very long string literal, which would surely exceed any limitations on how long a line or a string literal can be. The string literal alone exceeds 256 characters. It also has a character outside the Basic Multilingual Plane: 😂. Here's a double quote: \". Here are some escaped newlines:\n\n\nHere is a literal newline:\nCall me Ishmael. Some years ago—never mind how long precisely—having little or no money in my purse, and nothing particular to interest me on shore, I thought I would sail about a little and see the watery part of the world. It is a way I have of driving off the spleen and regulating the circulation. Whenever I find myself growing grim about the mouth; whenever it is a damp, drizzly November in my soul; whenever I find myself involuntarily pausing before coffin warehouses, and bringing up the rear of every funeral I meet; and especially whenever my hypos get such an upper hand of me, that it requires a strong moral principle to prevent me from deliberately stepping into the street, and methodically knocking peoples hats off—then, I account it high time to get to sea as soon as I can. This is my substitute for pistol and ball. With a philosophical flourish Cato throws himself upon his sword; I quietly take to the ship. There is nothing surprising in this. If they but knew it, almost all men in their degree, some time or other, cherish very nearly the same feelings towards the ocean with me."