All patches and comments are welcome. Please squash your changes to logical
commits before using git-format-patch and git-send-email to
patches@git.madduck.net.
If you'd read over the Git project's submission guidelines and adhered to them,
I'd be especially grateful.
3 from asyncio.base_events import BaseEventLoop
4 from concurrent.futures import Executor, ProcessPoolExecutor
5 from functools import partial
8 from pathlib import Path
11 Dict, Generic, Iterable, Iterator, List, Optional, Set, Tuple, TypeVar, Union
14 from attr import attrib, dataclass, Factory
18 from blib2to3.pytree import Node, Leaf, type_repr
19 from blib2to3 import pygram, pytree
20 from blib2to3.pgen2 import driver, token
21 from blib2to3.pgen2.parse import ParseError
23 __version__ = "18.3a1"
24 DEFAULT_LINE_LENGTH = 88
26 syms = pygram.python_symbols
33 LN = Union[Leaf, Node]
34 out = partial(click.secho, bold=True, err=True)
35 err = partial(click.secho, fg='red', err=True)
38 class NothingChanged(UserWarning):
39 """Raised by `format_file` when the reformatted code is the same as source."""
42 class CannotSplit(Exception):
43 """A readable split that fits the allotted line length is impossible.
45 Raised by `left_hand_split()` and `right_hand_split()`.
54 default=DEFAULT_LINE_LENGTH,
55 help='How many character per line to allow.',
61 help='If --fast given, skip temporary sanity checks. [default: --safe]',
63 @click.version_option(version=__version__)
67 type=click.Path(exists=True, file_okay=True, dir_okay=True, readable=True),
70 def main(ctx: click.Context, line_length: int, fast: bool, src: List[str]) -> None:
71 """The uncompromising code formatter."""
72 sources: List[Path] = []
76 sources.extend(gen_python_files_in_dir(p))
78 # if a file was explicitly given, we don't care about its extension
81 err(f'invalid path: {s}')
84 elif len(sources) == 1:
88 changed = format_file_in_place(p, line_length=line_length, fast=fast)
89 report.done(p, changed)
90 except Exception as exc:
91 report.failed(p, str(exc))
92 ctx.exit(report.return_code)
94 loop = asyncio.get_event_loop()
95 executor = ProcessPoolExecutor(max_workers=os.cpu_count())
98 return_code = loop.run_until_complete(
99 schedule_formatting(sources, line_length, fast, loop, executor)
103 ctx.exit(return_code)
106 async def schedule_formatting(
114 src: loop.run_in_executor(
115 executor, format_file_in_place, src, line_length, fast
119 await asyncio.wait(tasks.values())
122 for src, task in tasks.items():
124 report.failed(src, 'timed out, cancelling')
126 cancelled.append(task)
127 elif task.exception():
128 report.failed(src, str(task.exception()))
130 report.done(src, task.result())
132 await asyncio.wait(cancelled, timeout=2)
133 out('All done! ✨ 🍰 ✨')
134 click.echo(str(report))
135 return report.return_code
138 def format_file_in_place(src: Path, line_length: int, fast: bool) -> bool:
139 """Format the file and rewrite if changed. Return True if changed."""
141 contents, encoding = format_file(src, line_length=line_length, fast=fast)
142 except NothingChanged:
145 with open(src, "w", encoding=encoding) as f:
151 src: Path, line_length: int, fast: bool
152 ) -> Tuple[FileContent, Encoding]:
153 """Reformats a file and returns its contents and encoding."""
154 with tokenize.open(src) as src_buffer:
155 src_contents = src_buffer.read()
156 if src_contents.strip() == '':
157 raise NothingChanged(src)
159 dst_contents = format_str(src_contents, line_length=line_length)
160 if src_contents == dst_contents:
161 raise NothingChanged(src)
164 assert_equivalent(src_contents, dst_contents)
165 assert_stable(src_contents, dst_contents, line_length=line_length)
166 return dst_contents, src_buffer.encoding
169 def format_str(src_contents: str, line_length: int) -> FileContent:
170 """Reformats a string and returns new contents."""
171 src_node = lib2to3_parse(src_contents)
173 comments: List[Line] = []
174 lines = LineGenerator()
175 elt = EmptyLineTracker()
178 for current_line in lines.visit(src_node):
179 for _ in range(after):
180 dst_contents += str(empty_line)
181 before, after = elt.maybe_empty_lines(current_line)
182 for _ in range(before):
183 dst_contents += str(empty_line)
184 if not current_line.is_comment:
185 for comment in comments:
186 dst_contents += str(comment)
188 for line in split_line(current_line, line_length=line_length):
189 dst_contents += str(line)
191 comments.append(current_line)
192 for comment in comments:
193 dst_contents += str(comment)
197 def lib2to3_parse(src_txt: str) -> Node:
198 """Given a string with source, return the lib2to3 Node."""
199 grammar = pygram.python_grammar_no_print_statement
200 drv = driver.Driver(grammar, pytree.convert)
201 if src_txt[-1] != '\n':
202 nl = '\r\n' if '\r\n' in src_txt[:1024] else '\n'
205 result = drv.parse_string(src_txt, True)
206 except ParseError as pe:
207 lineno, column = pe.context[1]
208 lines = src_txt.splitlines()
210 faulty_line = lines[lineno - 1]
212 faulty_line = "<line number missing in source>"
213 raise ValueError(f"Cannot parse: {lineno}:{column}: {faulty_line}") from None
215 if isinstance(result, Leaf):
216 result = Node(syms.file_input, [result])
220 def lib2to3_unparse(node: Node) -> str:
221 """Given a lib2to3 node, return its string representation."""
229 class Visitor(Generic[T]):
230 """Basic lib2to3 visitor that yields things on visiting."""
232 def visit(self, node: LN) -> Iterator[T]:
234 name = token.tok_name[node.type]
236 name = type_repr(node.type)
237 yield from getattr(self, f'visit_{name}', self.visit_default)(node)
239 def visit_default(self, node: LN) -> Iterator[T]:
240 if isinstance(node, Node):
241 for child in node.children:
242 yield from self.visit(child)
246 class DebugVisitor(Visitor[T]):
247 tree_depth: int = attrib(default=0)
249 def visit_default(self, node: LN) -> Iterator[T]:
250 indent = ' ' * (2 * self.tree_depth)
251 if isinstance(node, Node):
252 _type = type_repr(node.type)
253 out(f'{indent}{_type}', fg='yellow')
255 for child in node.children:
256 yield from self.visit(child)
259 out(f'{indent}/{_type}', fg='yellow', bold=False)
261 _type = token.tok_name.get(node.type, str(node.type))
262 out(f'{indent}{_type}', fg='blue', nl=False)
264 # We don't have to handle prefixes for `Node` objects since
265 # that delegates to the first child anyway.
266 out(f' {node.prefix!r}', fg='green', bold=False, nl=False)
267 out(f' {node.value!r}', fg='blue', bold=False)
270 KEYWORDS = set(keyword.kwlist)
271 WHITESPACE = {token.DEDENT, token.INDENT, token.NEWLINE}
272 FLOW_CONTROL = {'return', 'raise', 'break', 'continue'}
283 STANDALONE_COMMENT = 153
284 LOGIC_OPERATORS = {'and', 'or'}
307 COMPREHENSION_PRIORITY = 20
311 COMPARATOR_PRIORITY = 3
316 class BracketTracker:
317 depth: int = attrib(default=0)
318 bracket_match: Dict[Tuple[Depth, NodeType], Leaf] = attrib(default=Factory(dict))
319 delimiters: Dict[LeafID, Priority] = attrib(default=Factory(dict))
320 previous: Optional[Leaf] = attrib(default=None)
322 def mark(self, leaf: Leaf) -> None:
323 if leaf.type == token.COMMENT:
326 if leaf.type in CLOSING_BRACKETS:
328 opening_bracket = self.bracket_match.pop((self.depth, leaf.type))
329 leaf.opening_bracket = opening_bracket # type: ignore
330 leaf.bracket_depth = self.depth # type: ignore
332 delim = is_delimiter(leaf)
334 self.delimiters[id(leaf)] = delim
335 elif self.previous is not None:
336 if leaf.type == token.STRING and self.previous.type == token.STRING:
337 self.delimiters[id(self.previous)] = STRING_PRIORITY
339 leaf.type == token.NAME and
340 leaf.value == 'for' and
342 leaf.parent.type in {syms.comp_for, syms.old_comp_for}
344 self.delimiters[id(self.previous)] = COMPREHENSION_PRIORITY
346 leaf.type == token.NAME and
347 leaf.value == 'if' and
349 leaf.parent.type in {syms.comp_if, syms.old_comp_if}
351 self.delimiters[id(self.previous)] = COMPREHENSION_PRIORITY
352 if leaf.type in OPENING_BRACKETS:
353 self.bracket_match[self.depth, BRACKET[leaf.type]] = leaf
357 def any_open_brackets(self) -> bool:
358 """Returns True if there is an yet unmatched open bracket on the line."""
359 return bool(self.bracket_match)
361 def max_priority(self, exclude: Iterable[LeafID] = ()) -> int:
362 """Returns the highest priority of a delimiter found on the line.
364 Values are consistent with what `is_delimiter()` returns.
366 return max(v for k, v in self.delimiters.items() if k not in exclude)
371 depth: int = attrib(default=0)
372 leaves: List[Leaf] = attrib(default=Factory(list))
373 comments: Dict[LeafID, Leaf] = attrib(default=Factory(dict))
374 bracket_tracker: BracketTracker = attrib(default=Factory(BracketTracker))
375 inside_brackets: bool = attrib(default=False)
377 def append(self, leaf: Leaf, preformatted: bool = False) -> None:
378 has_value = leaf.value.strip()
382 if self.leaves and not preformatted:
383 # Note: at this point leaf.prefix should be empty except for
384 # imports, for which we only preserve newlines.
385 leaf.prefix += whitespace(leaf)
386 if self.inside_brackets or not preformatted:
387 self.bracket_tracker.mark(leaf)
388 self.maybe_remove_trailing_comma(leaf)
389 if self.maybe_adapt_standalone_comment(leaf):
392 if not self.append_comment(leaf):
393 self.leaves.append(leaf)
396 def is_comment(self) -> bool:
397 return bool(self) and self.leaves[0].type == STANDALONE_COMMENT
400 def is_decorator(self) -> bool:
401 return bool(self) and self.leaves[0].type == token.AT
404 def is_import(self) -> bool:
405 return bool(self) and is_import(self.leaves[0])
408 def is_class(self) -> bool:
411 self.leaves[0].type == token.NAME and
412 self.leaves[0].value == 'class'
416 def is_def(self) -> bool:
417 """Also returns True for async defs."""
419 first_leaf = self.leaves[0]
424 second_leaf: Optional[Leaf] = self.leaves[1]
428 (first_leaf.type == token.NAME and first_leaf.value == 'def') or
430 first_leaf.type == token.NAME and
431 first_leaf.value == 'async' and
432 second_leaf is not None and
433 second_leaf.type == token.NAME and
434 second_leaf.value == 'def'
439 def is_flow_control(self) -> bool:
442 self.leaves[0].type == token.NAME and
443 self.leaves[0].value in FLOW_CONTROL
447 def is_yield(self) -> bool:
450 self.leaves[0].type == token.NAME and
451 self.leaves[0].value == 'yield'
454 def maybe_remove_trailing_comma(self, closing: Leaf) -> bool:
457 self.leaves[-1].type == token.COMMA and
458 closing.type in CLOSING_BRACKETS
462 if closing.type == token.RSQB or closing.type == token.RBRACE:
466 # For parens let's check if it's safe to remove the comma. If the
467 # trailing one is the only one, we might mistakenly change a tuple
468 # into a different type by removing the comma.
469 depth = closing.bracket_depth + 1 # type: ignore
471 opening = closing.opening_bracket # type: ignore
472 for _opening_index, leaf in enumerate(self.leaves):
479 for leaf in self.leaves[_opening_index + 1:]:
483 bracket_depth = leaf.bracket_depth # type: ignore
484 if bracket_depth == depth and leaf.type == token.COMMA:
492 def maybe_adapt_standalone_comment(self, comment: Leaf) -> bool:
493 """Hack a standalone comment to act as a trailing comment for line splitting.
495 If this line has brackets and a standalone `comment`, we need to adapt
496 it to be able to still reformat the line.
498 This is not perfect, the line to which the standalone comment gets
499 appended will appear "too long" when splitting.
502 comment.type == STANDALONE_COMMENT and
503 self.bracket_tracker.any_open_brackets()
507 comment.type = token.COMMENT
508 comment.prefix = '\n' + ' ' * (self.depth + 1)
509 return self.append_comment(comment)
511 def append_comment(self, comment: Leaf) -> bool:
512 if comment.type != token.COMMENT:
516 after = id(self.last_non_delimiter())
518 comment.type = STANDALONE_COMMENT
523 if after in self.comments:
524 self.comments[after].value += str(comment)
526 self.comments[after] = comment
529 def last_non_delimiter(self) -> Leaf:
530 for i in range(len(self.leaves)):
531 last = self.leaves[-i - 1]
532 if not is_delimiter(last):
535 raise LookupError("No non-delimiters found")
537 def __str__(self) -> str:
541 indent = ' ' * self.depth
542 leaves = iter(self.leaves)
544 res = f'{first.prefix}{indent}{first.value}'
547 for comment in self.comments.values():
551 def __bool__(self) -> bool:
552 return bool(self.leaves or self.comments)
556 class EmptyLineTracker:
557 """Provides a stateful method that returns the number of potential extra
558 empty lines needed before and after the currently processed line.
560 Note: this tracker works on lines that haven't been split yet.
562 previous_line: Optional[Line] = attrib(default=None)
563 previous_after: int = attrib(default=0)
564 previous_defs: List[int] = attrib(default=Factory(list))
566 def maybe_empty_lines(self, current_line: Line) -> Tuple[int, int]:
567 """Returns the number of extra empty lines before and after the `current_line`.
569 This is for separating `def`, `async def` and `class` with extra empty lines
570 (two on module-level), as well as providing an extra empty line after flow
571 control keywords to make them more prominent.
573 before, after = self._maybe_empty_lines(current_line)
574 self.previous_after = after
575 self.previous_line = current_line
578 def _maybe_empty_lines(self, current_line: Line) -> Tuple[int, int]:
580 depth = current_line.depth
581 while self.previous_defs and self.previous_defs[-1] >= depth:
582 self.previous_defs.pop()
583 before = (1 if depth else 2) - self.previous_after
584 is_decorator = current_line.is_decorator
585 if is_decorator or current_line.is_def or current_line.is_class:
587 self.previous_defs.append(depth)
588 if self.previous_line is None:
589 # Don't insert empty lines before the first line in the file.
592 if self.previous_line and self.previous_line.is_decorator:
593 # Don't insert empty lines between decorators.
597 if current_line.depth:
599 newlines -= self.previous_after
602 if current_line.is_flow_control:
606 self.previous_line and
607 self.previous_line.is_import and
608 not current_line.is_import and
609 depth == self.previous_line.depth
611 return (before or 1), 0
614 self.previous_line and
615 self.previous_line.is_yield and
616 (not current_line.is_yield or depth != self.previous_line.depth)
618 return (before or 1), 0
624 class LineGenerator(Visitor[Line]):
625 """Generates reformatted Line objects. Empty lines are not emitted.
627 Note: destroys the tree it's visiting by mutating prefixes of its leaves
628 in ways that will no longer stringify to valid Python code on the tree.
630 current_line: Line = attrib(default=Factory(Line))
631 standalone_comments: List[Leaf] = attrib(default=Factory(list))
633 def line(self, indent: int = 0) -> Iterator[Line]:
636 If the line is empty, only emit if it makes sense.
637 If the line is too long, split it first and then generate.
639 If any lines were generated, set up a new current_line.
641 if not self.current_line:
642 self.current_line.depth += indent
643 return # Line is empty, don't emit. Creating a new one unnecessary.
645 complete_line = self.current_line
646 self.current_line = Line(depth=complete_line.depth + indent)
649 def visit_default(self, node: LN) -> Iterator[Line]:
650 if isinstance(node, Leaf):
651 for comment in generate_comments(node):
652 if self.current_line.bracket_tracker.any_open_brackets():
653 # any comment within brackets is subject to splitting
654 self.current_line.append(comment)
655 elif comment.type == token.COMMENT:
656 # regular trailing comment
657 self.current_line.append(comment)
658 yield from self.line()
661 # regular standalone comment, to be processed later (see
662 # docstring in `generate_comments()`
663 self.standalone_comments.append(comment)
664 normalize_prefix(node)
665 if node.type not in WHITESPACE:
666 for comment in self.standalone_comments:
667 yield from self.line()
669 self.current_line.append(comment)
670 yield from self.line()
672 self.standalone_comments = []
673 self.current_line.append(node)
674 yield from super().visit_default(node)
676 def visit_suite(self, node: Node) -> Iterator[Line]:
677 """Body of a statement after a colon."""
678 children = iter(node.children)
679 # Process newline before indenting. It might contain an inline
680 # comment that should go right after the colon.
681 newline = next(children)
682 yield from self.visit(newline)
683 yield from self.line(+1)
685 for child in children:
686 yield from self.visit(child)
688 yield from self.line(-1)
690 def visit_stmt(self, node: Node, keywords: Set[str]) -> Iterator[Line]:
691 """Visit a statement.
693 The relevant Python language keywords for this statement are NAME leaves
696 for child in node.children:
697 if child.type == token.NAME and child.value in keywords: # type: ignore
698 yield from self.line()
700 yield from self.visit(child)
702 def visit_simple_stmt(self, node: Node) -> Iterator[Line]:
703 """A statement without nested statements."""
704 is_suite_like = node.parent and node.parent.type in STATEMENT
706 yield from self.line(+1)
707 yield from self.visit_default(node)
708 yield from self.line(-1)
711 yield from self.line()
712 yield from self.visit_default(node)
714 def visit_async_stmt(self, node: Node) -> Iterator[Line]:
715 yield from self.line()
717 children = iter(node.children)
718 for child in children:
719 yield from self.visit(child)
721 if child.type == token.NAME and child.value == 'async': # type: ignore
724 internal_stmt = next(children)
725 for child in internal_stmt.children:
726 yield from self.visit(child)
728 def visit_decorators(self, node: Node) -> Iterator[Line]:
729 for child in node.children:
730 yield from self.line()
731 yield from self.visit(child)
733 def visit_SEMI(self, leaf: Leaf) -> Iterator[Line]:
734 yield from self.line()
736 def visit_ENDMARKER(self, leaf: Leaf) -> Iterator[Line]:
737 yield from self.visit_default(leaf)
738 yield from self.line()
740 def __attrs_post_init__(self) -> None:
741 """You are in a twisty little maze of passages."""
743 self.visit_if_stmt = partial(v, keywords={'if', 'else', 'elif'})
744 self.visit_while_stmt = partial(v, keywords={'while', 'else'})
745 self.visit_for_stmt = partial(v, keywords={'for', 'else'})
746 self.visit_try_stmt = partial(v, keywords={'try', 'except', 'else', 'finally'})
747 self.visit_except_clause = partial(v, keywords={'except'})
748 self.visit_funcdef = partial(v, keywords={'def'})
749 self.visit_with_stmt = partial(v, keywords={'with'})
750 self.visit_classdef = partial(v, keywords={'class'})
751 self.visit_async_funcdef = self.visit_async_stmt
752 self.visit_decorated = self.visit_decorators
755 BRACKET = {token.LPAR: token.RPAR, token.LSQB: token.RSQB, token.LBRACE: token.RBRACE}
756 OPENING_BRACKETS = set(BRACKET.keys())
757 CLOSING_BRACKETS = set(BRACKET.values())
758 BRACKETS = OPENING_BRACKETS | CLOSING_BRACKETS
761 def whitespace(leaf: Leaf) -> str:
762 """Return whitespace prefix if needed for the given `leaf`."""
778 if t == token.COMMENT:
781 if t == STANDALONE_COMMENT:
784 assert p is not None, f"INTERNAL ERROR: hand-made leaf without parent: {leaf!r}"
785 if p.type in {syms.parameters, syms.arglist}:
786 # untyped function signatures or calls
790 prev = leaf.prev_sibling
791 if not prev or prev.type != token.COMMA:
794 if p.type == syms.varargslist:
799 prev = leaf.prev_sibling
800 if prev and prev.type != token.COMMA:
803 elif p.type == syms.typedargslist:
804 # typed function signatures
805 prev = leaf.prev_sibling
810 if prev.type != syms.tname:
813 elif prev.type == token.EQUAL:
814 # A bit hacky: if the equal sign has whitespace, it means we
815 # previously found it's a typed argument. So, we're using that, too.
818 elif prev.type != token.COMMA:
821 elif p.type == syms.tname:
823 prev = leaf.prev_sibling
825 prevp = preceding_leaf(p)
826 if not prevp or prevp.type != token.COMMA:
829 elif p.type == syms.trailer:
830 # attributes and calls
831 if t == token.LPAR or t == token.RPAR:
834 prev = leaf.prev_sibling
837 prevp = preceding_leaf(p)
838 if not prevp or prevp.type != token.NUMBER:
841 elif t == token.LSQB:
844 elif prev.type != token.COMMA:
847 elif p.type == syms.argument:
852 prev = leaf.prev_sibling
854 prevp = preceding_leaf(p)
855 if not prevp or prevp.type == token.LPAR:
858 elif prev.type == token.EQUAL or prev.type == token.DOUBLESTAR:
861 elif p.type == syms.decorator:
865 elif p.type == syms.dotted_name:
866 prev = leaf.prev_sibling
870 prevp = preceding_leaf(p)
871 if not prevp or prevp.type == token.AT or prevp.type == token.DOT:
874 elif p.type == syms.classdef:
878 prev = leaf.prev_sibling
879 if prev and prev.type == token.LPAR:
882 elif p.type == syms.subscript:
887 prev = leaf.prev_sibling
888 if not prev or prev.type == token.COLON:
904 # various arithmetic and logic expressions
905 prev = leaf.prev_sibling
907 prevp = preceding_leaf(p)
908 if not prevp or prevp.type in OPENING_BRACKETS:
911 if prevp.type == token.EQUAL:
912 if prevp.parent and prevp.parent.type in {
913 syms.varargslist, syms.parameters, syms.arglist, syms.argument
919 elif p.type == syms.atom:
920 if t in CLOSING_BRACKETS:
923 prev = leaf.prev_sibling
925 prevp = preceding_leaf(p)
929 if prevp.type in OPENING_BRACKETS:
932 if prevp.type == token.EQUAL:
933 if prevp.parent and prevp.parent.type in {
934 syms.varargslist, syms.parameters, syms.arglist, syms.argument
938 if prevp.type == token.DOUBLESTAR:
939 if prevp.parent and prevp.parent.type in {
940 syms.varargslist, syms.parameters, syms.arglist, syms.dictsetmaker
944 elif prev.type in OPENING_BRACKETS:
948 # dots, but not the first one.
952 p.type == syms.listmaker or
953 p.type == syms.testlist_gexp or
954 p.type == syms.subscriptlist
956 # list interior, including unpacking
957 prev = leaf.prev_sibling
961 elif p.type == syms.dictsetmaker:
962 # dict and set interior, including unpacking
963 prev = leaf.prev_sibling
967 if prev.type == token.DOUBLESTAR:
970 elif p.type == syms.factor or p.type == syms.star_expr:
972 prev = leaf.prev_sibling
974 prevp = preceding_leaf(p)
975 if not prevp or prevp.type in OPENING_BRACKETS:
978 prevp_parent = prevp.parent
979 assert prevp_parent is not None
980 if prevp.type == token.COLON and prevp_parent.type in {
981 syms.subscript, syms.sliceop
985 elif prevp.type == token.EQUAL and prevp_parent.type == syms.argument:
988 elif t == token.NAME or t == token.NUMBER:
991 elif p.type == syms.import_from:
993 prev = leaf.prev_sibling
994 if prev and prev.type == token.DOT:
997 elif t == token.NAME:
1001 prev = leaf.prev_sibling
1002 if prev and prev.type == token.DOT:
1005 elif p.type == syms.sliceop:
1011 def preceding_leaf(node: Optional[LN]) -> Optional[Leaf]:
1012 """Returns the first leaf that precedes `node`, if any."""
1014 res = node.prev_sibling
1016 if isinstance(res, Leaf):
1020 return list(res.leaves())[-1]
1029 def is_delimiter(leaf: Leaf) -> int:
1030 """Returns the priority of the `leaf` delimiter. Returns 0 if not delimiter.
1032 Higher numbers are higher priority.
1034 if leaf.type == token.COMMA:
1035 return COMMA_PRIORITY
1037 if leaf.type == token.NAME and leaf.value in LOGIC_OPERATORS:
1038 return LOGIC_PRIORITY
1040 if leaf.type in COMPARATORS:
1041 return COMPARATOR_PRIORITY
1044 leaf.type in MATH_OPERATORS and
1046 leaf.parent.type not in {syms.factor, syms.star_expr}
1048 return MATH_PRIORITY
1053 def generate_comments(leaf: Leaf) -> Iterator[Leaf]:
1054 """Cleans the prefix of the `leaf` and generates comments from it, if any.
1056 Comments in lib2to3 are shoved into the whitespace prefix. This happens
1057 in `pgen2/driver.py:Driver.parse_tokens()`. This was a brilliant implementation
1058 move because it does away with modifying the grammar to include all the
1059 possible places in which comments can be placed.
1061 The sad consequence for us though is that comments don't "belong" anywhere.
1062 This is why this function generates simple parentless Leaf objects for
1063 comments. We simply don't know what the correct parent should be.
1065 No matter though, we can live without this. We really only need to
1066 differentiate between inline and standalone comments. The latter don't
1067 share the line with any code.
1069 Inline comments are emitted as regular token.COMMENT leaves. Standalone
1070 are emitted with a fake STANDALONE_COMMENT token identifier.
1075 if '#' not in leaf.prefix:
1078 before_comment, content = leaf.prefix.split('#', 1)
1079 content = content.rstrip()
1080 if content and (content[0] not in {' ', '!', '#'}):
1081 content = ' ' + content
1082 is_standalone_comment = (
1083 '\n' in before_comment or '\n' in content or leaf.type == token.DEDENT
1085 if not is_standalone_comment:
1086 # simple trailing comment
1087 yield Leaf(token.COMMENT, value='#' + content)
1090 for line in ('#' + content).split('\n'):
1091 line = line.lstrip()
1092 if not line.startswith('#'):
1095 yield Leaf(STANDALONE_COMMENT, line)
1098 def split_line(line: Line, line_length: int, inner: bool = False) -> Iterator[Line]:
1099 """Splits a `line` into potentially many lines.
1101 They should fit in the allotted `line_length` but might not be able to.
1102 `inner` signifies that there were a pair of brackets somewhere around the
1103 current `line`, possibly transitively. This means we can fallback to splitting
1104 by delimiters if the LHS/RHS don't yield any results.
1106 line_str = str(line).strip('\n')
1107 if len(line_str) <= line_length and '\n' not in line_str:
1112 split_funcs = [left_hand_split]
1113 elif line.inside_brackets:
1114 split_funcs = [delimiter_split]
1115 if '\n' not in line_str:
1116 # Only attempt RHS if we don't have multiline strings or comments
1118 split_funcs.append(right_hand_split)
1120 split_funcs = [right_hand_split]
1121 for split_func in split_funcs:
1122 # We are accumulating lines in `result` because we might want to abort
1123 # mission and return the original line in the end, or attempt a different
1125 result: List[Line] = []
1127 for l in split_func(line):
1128 if str(l).strip('\n') == line_str:
1129 raise CannotSplit("Split function returned an unchanged result")
1131 result.extend(split_line(l, line_length=line_length, inner=True))
1132 except CannotSplit as cs:
1143 def left_hand_split(line: Line) -> Iterator[Line]:
1144 """Split line into many lines, starting with the first matching bracket pair.
1146 Note: this usually looks weird, only use this for function definitions.
1147 Prefer RHS otherwise.
1149 head = Line(depth=line.depth)
1150 body = Line(depth=line.depth + 1, inside_brackets=True)
1151 tail = Line(depth=line.depth)
1152 tail_leaves: List[Leaf] = []
1153 body_leaves: List[Leaf] = []
1154 head_leaves: List[Leaf] = []
1155 current_leaves = head_leaves
1156 matching_bracket = None
1157 for leaf in line.leaves:
1159 current_leaves is body_leaves and
1160 leaf.type in CLOSING_BRACKETS and
1161 leaf.opening_bracket is matching_bracket # type: ignore
1163 current_leaves = tail_leaves
1164 current_leaves.append(leaf)
1165 if current_leaves is head_leaves:
1166 if leaf.type in OPENING_BRACKETS:
1167 matching_bracket = leaf
1168 current_leaves = body_leaves
1169 # Since body is a new indent level, remove spurious leading whitespace.
1171 normalize_prefix(body_leaves[0])
1172 # Build the new lines.
1173 for result, leaves in (
1174 (head, head_leaves), (body, body_leaves), (tail, tail_leaves)
1177 result.append(leaf, preformatted=True)
1178 comment_after = line.comments.get(id(leaf))
1180 result.append(comment_after, preformatted=True)
1181 # Check if the split succeeded.
1182 tail_len = len(str(tail))
1185 raise CannotSplit("Splitting brackets produced the same line")
1189 f"Splitting brackets on an empty body to save "
1190 f"{tail_len} characters is not worth it"
1193 for result in (head, body, tail):
1198 def right_hand_split(line: Line) -> Iterator[Line]:
1199 """Split line into many lines, starting with the last matching bracket pair."""
1200 head = Line(depth=line.depth)
1201 body = Line(depth=line.depth + 1, inside_brackets=True)
1202 tail = Line(depth=line.depth)
1203 tail_leaves: List[Leaf] = []
1204 body_leaves: List[Leaf] = []
1205 head_leaves: List[Leaf] = []
1206 current_leaves = tail_leaves
1207 opening_bracket = None
1208 for leaf in reversed(line.leaves):
1209 if current_leaves is body_leaves:
1210 if leaf is opening_bracket:
1211 current_leaves = head_leaves
1212 current_leaves.append(leaf)
1213 if current_leaves is tail_leaves:
1214 if leaf.type in CLOSING_BRACKETS:
1215 opening_bracket = leaf.opening_bracket # type: ignore
1216 current_leaves = body_leaves
1217 tail_leaves.reverse()
1218 body_leaves.reverse()
1219 head_leaves.reverse()
1220 # Since body is a new indent level, remove spurious leading whitespace.
1222 normalize_prefix(body_leaves[0])
1223 # Build the new lines.
1224 for result, leaves in (
1225 (head, head_leaves), (body, body_leaves), (tail, tail_leaves)
1228 result.append(leaf, preformatted=True)
1229 comment_after = line.comments.get(id(leaf))
1231 result.append(comment_after, preformatted=True)
1232 # Check if the split succeeded.
1233 tail_len = len(str(tail).strip('\n'))
1236 raise CannotSplit("Splitting brackets produced the same line")
1240 f"Splitting brackets on an empty body to save "
1241 f"{tail_len} characters is not worth it"
1244 for result in (head, body, tail):
1249 def delimiter_split(line: Line) -> Iterator[Line]:
1250 """Split according to delimiters of the highest priority.
1252 This kind of split doesn't increase indentation.
1255 last_leaf = line.leaves[-1]
1257 raise CannotSplit("Line empty")
1259 delimiters = line.bracket_tracker.delimiters
1261 delimiter_priority = line.bracket_tracker.max_priority(exclude={id(last_leaf)})
1263 raise CannotSplit("No delimiters found")
1265 current_line = Line(depth=line.depth, inside_brackets=line.inside_brackets)
1266 for leaf in line.leaves:
1267 current_line.append(leaf, preformatted=True)
1268 comment_after = line.comments.get(id(leaf))
1270 current_line.append(comment_after, preformatted=True)
1271 leaf_priority = delimiters.get(id(leaf))
1272 if leaf_priority == delimiter_priority:
1273 normalize_prefix(current_line.leaves[0])
1276 current_line = Line(depth=line.depth, inside_brackets=line.inside_brackets)
1279 delimiter_priority == COMMA_PRIORITY and
1280 current_line.leaves[-1].type != token.COMMA
1282 current_line.append(Leaf(token.COMMA, ','))
1283 normalize_prefix(current_line.leaves[0])
1287 def is_import(leaf: Leaf) -> bool:
1288 """Returns True if the given leaf starts an import statement."""
1295 (v == 'import' and p and p.type == syms.import_name) or
1296 (v == 'from' and p and p.type == syms.import_from)
1301 def normalize_prefix(leaf: Leaf) -> None:
1302 """Leave existing extra newlines for imports. Remove everything else."""
1304 spl = leaf.prefix.split('#', 1)
1305 nl_count = spl[0].count('\n')
1307 # Skip one newline since it was for a standalone comment.
1309 leaf.prefix = '\n' * nl_count
1315 PYTHON_EXTENSIONS = {'.py'}
1316 BLACKLISTED_DIRECTORIES = {
1317 'build', 'buck-out', 'dist', '_build', '.git', '.hg', '.mypy_cache', '.tox', '.venv'
1321 def gen_python_files_in_dir(path: Path) -> Iterator[Path]:
1322 for child in path.iterdir():
1324 if child.name in BLACKLISTED_DIRECTORIES:
1327 yield from gen_python_files_in_dir(child)
1329 elif child.suffix in PYTHON_EXTENSIONS:
1335 """Provides a reformatting counter."""
1336 change_count: int = attrib(default=0)
1337 same_count: int = attrib(default=0)
1338 failure_count: int = attrib(default=0)
1340 def done(self, src: Path, changed: bool) -> None:
1341 """Increment the counter for successful reformatting. Write out a message."""
1343 out(f'reformatted {src}')
1344 self.change_count += 1
1346 out(f'{src} already well formatted, good job.', bold=False)
1347 self.same_count += 1
1349 def failed(self, src: Path, message: str) -> None:
1350 """Increment the counter for failed reformatting. Write out a message."""
1351 err(f'error: cannot format {src}: {message}')
1352 self.failure_count += 1
1355 def return_code(self) -> int:
1356 """Which return code should the app use considering the current state."""
1357 return 1 if self.failure_count else 0
1359 def __str__(self) -> str:
1360 """A color report of the current state.
1362 Use `click.unstyle` to remove colors.
1365 if self.change_count:
1366 s = 's' if self.change_count > 1 else ''
1368 click.style(f'{self.change_count} file{s} reformatted', bold=True)
1371 s = 's' if self.same_count > 1 else ''
1372 report.append(f'{self.same_count} file{s} left unchanged')
1373 if self.failure_count:
1374 s = 's' if self.failure_count > 1 else ''
1377 f'{self.failure_count} file{s} failed to reformat', fg='red'
1380 return ', '.join(report) + '.'
1383 def assert_equivalent(src: str, dst: str) -> None:
1384 """Raises AssertionError if `src` and `dst` aren't equivalent.
1386 This is a temporary sanity check until Black becomes stable.
1392 def _v(node: ast.AST, depth: int = 0) -> Iterator[str]:
1393 """Simple visitor generating strings to compare ASTs by content."""
1394 yield f"{' ' * depth}{node.__class__.__name__}("
1396 for field in sorted(node._fields):
1398 value = getattr(node, field)
1399 except AttributeError:
1402 yield f"{' ' * (depth+1)}{field}="
1404 if isinstance(value, list):
1406 if isinstance(item, ast.AST):
1407 yield from _v(item, depth + 2)
1409 elif isinstance(value, ast.AST):
1410 yield from _v(value, depth + 2)
1413 yield f"{' ' * (depth+2)}{value!r}, # {value.__class__.__name__}"
1415 yield f"{' ' * depth}) # /{node.__class__.__name__}"
1418 src_ast = ast.parse(src)
1419 except Exception as exc:
1420 raise AssertionError(f"cannot parse source: {exc}") from None
1423 dst_ast = ast.parse(dst)
1424 except Exception as exc:
1425 log = dump_to_file(''.join(traceback.format_tb(exc.__traceback__)), dst)
1426 raise AssertionError(
1427 f"INTERNAL ERROR: Black produced invalid code: {exc}. "
1428 f"Please report a bug on https://github.com/ambv/black/issues. "
1429 f"This invalid output might be helpful: {log}",
1432 src_ast_str = '\n'.join(_v(src_ast))
1433 dst_ast_str = '\n'.join(_v(dst_ast))
1434 if src_ast_str != dst_ast_str:
1435 log = dump_to_file(diff(src_ast_str, dst_ast_str, 'src', 'dst'))
1436 raise AssertionError(
1437 f"INTERNAL ERROR: Black produced code that is not equivalent to "
1439 f"Please report a bug on https://github.com/ambv/black/issues. "
1440 f"This diff might be helpful: {log}",
1444 def assert_stable(src: str, dst: str, line_length: int) -> None:
1445 """Raises AssertionError if `dst` reformats differently the second time.
1447 This is a temporary sanity check until Black becomes stable.
1449 newdst = format_str(dst, line_length=line_length)
1452 diff(src, dst, 'source', 'first pass'),
1453 diff(dst, newdst, 'first pass', 'second pass'),
1455 raise AssertionError(
1456 f"INTERNAL ERROR: Black produced different code on the second pass "
1457 f"of the formatter. "
1458 f"Please report a bug on https://github.com/ambv/black/issues. "
1459 f"This diff might be helpful: {log}",
1463 def dump_to_file(*output: str) -> str:
1464 """Dumps `output` to a temporary file. Returns path to the file."""
1467 with tempfile.NamedTemporaryFile(
1468 mode='w', prefix='blk_', suffix='.log', delete=False
1470 for lines in output:
1476 def diff(a: str, b: str, a_name: str, b_name: str) -> str:
1477 """Returns a udiff string between strings `a` and `b`."""
1480 a_lines = [line + '\n' for line in a.split('\n')]
1481 b_lines = [line + '\n' for line in b.split('\n')]
1483 difflib.unified_diff(a_lines, b_lines, fromfile=a_name, tofile=b_name, n=5)
1487 if __name__ == '__main__':