Skip to content

Commit 53c23e6

Browse files
Support files with type comment syntax errors (#3594)
1 parent dba3c26 commit 53c23e6

File tree

4 files changed

+39
-7
lines changed

4 files changed

+39
-7
lines changed

CHANGES.md

+2
Original file line numberDiff line numberDiff line change
@@ -38,6 +38,8 @@
3838

3939
<!-- Changes to the parser or to version autodetection -->
4040

41+
- Added support for formatting files with invalid type comments (#3594)
42+
4143
### Performance
4244

4345
<!-- Changes that improve Black's performance. -->

src/black/parsing.py

+19-7
Original file line numberDiff line numberDiff line change
@@ -148,24 +148,29 @@ def lib2to3_unparse(node: Node) -> str:
148148

149149

150150
def parse_single_version(
151-
src: str, version: Tuple[int, int]
151+
src: str, version: Tuple[int, int], *, type_comments: bool
152152
) -> Union[ast.AST, ast3.AST]:
153153
filename = "<unknown>"
154154
# typed-ast is needed because of feature version limitations in the builtin ast 3.8>
155155
if sys.version_info >= (3, 8) and version >= (3,):
156-
return ast.parse(src, filename, feature_version=version, type_comments=True)
156+
return ast.parse(
157+
src, filename, feature_version=version, type_comments=type_comments
158+
)
157159

158160
if _IS_PYPY:
159161
# PyPy 3.7 doesn't support type comment tracking which is not ideal, but there's
160162
# not much we can do as typed-ast won't work either.
161163
if sys.version_info >= (3, 8):
162-
return ast3.parse(src, filename, type_comments=True)
164+
return ast3.parse(src, filename, type_comments=type_comments)
163165
else:
164166
return ast3.parse(src, filename)
165167
else:
166-
# Typed-ast is guaranteed to be used here and automatically tracks type
167-
# comments separately.
168-
return ast3.parse(src, filename, feature_version=version[1])
168+
if type_comments:
169+
# Typed-ast is guaranteed to be used here and automatically tracks type
170+
# comments separately.
171+
return ast3.parse(src, filename, feature_version=version[1])
172+
else:
173+
return ast.parse(src, filename)
169174

170175

171176
def parse_ast(src: str) -> Union[ast.AST, ast3.AST]:
@@ -175,11 +180,18 @@ def parse_ast(src: str) -> Union[ast.AST, ast3.AST]:
175180
first_error = ""
176181
for version in sorted(versions, reverse=True):
177182
try:
178-
return parse_single_version(src, version)
183+
return parse_single_version(src, version, type_comments=True)
179184
except SyntaxError as e:
180185
if not first_error:
181186
first_error = str(e)
182187

188+
# Try to parse without type comments
189+
for version in sorted(versions, reverse=True):
190+
try:
191+
return parse_single_version(src, version, type_comments=False)
192+
except SyntaxError:
193+
pass
194+
183195
raise SyntaxError(first_error)
184196

185197

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,11 @@
1+
def foo(
2+
# type: Foo
3+
x): pass
4+
5+
# output
6+
7+
def foo(
8+
# type: Foo
9+
x,
10+
):
11+
pass

tests/test_format.py

+7
Original file line numberDiff line numberDiff line change
@@ -196,3 +196,10 @@ def test_power_op_newline() -> None:
196196
# requires line_length=0
197197
source, expected = read_data("miscellaneous", "power_op_newline")
198198
assert_format(source, expected, mode=black.Mode(line_length=0))
199+
200+
201+
def test_type_comment_syntax_error() -> None:
202+
"""Test that black is able to format python code with type comment syntax errors."""
203+
source, expected = read_data("type_comments", "type_comment_syntax_error")
204+
assert_format(source, expected)
205+
black.assert_equivalent(source, expected)

0 commit comments

Comments
 (0)