]> git.madduck.net Git - etc/vim.git/blob - src/black/linegen.py

madduck's git repository

Every one of the projects in this repository is available at the canonical URL git://git.madduck.net/madduck/pub/<projectpath> — see each project's metadata for the exact URL.

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.

SSH access, as well as push access can be individually arranged.

If you use my repositories frequently, consider adding the following snippet to ~/.gitconfig and using the third clone URL listed for each project:

[url "git://git.madduck.net/madduck/"]
  insteadOf = madduck:

Doc: updating url link (#3739)
[etc/vim.git] / src / black / linegen.py
1 """
2 Generating lines of code.
3 """
4 import sys
5 from dataclasses import replace
6 from enum import Enum, auto
7 from functools import partial, wraps
8 from typing import Collection, Iterator, List, Optional, Set, Union, cast
9
10 from black.brackets import (
11     COMMA_PRIORITY,
12     DOT_PRIORITY,
13     get_leaves_inside_matching_brackets,
14     max_delimiter_priority_in_atom,
15 )
16 from black.comments import FMT_OFF, generate_comments, list_comments
17 from black.lines import (
18     Line,
19     RHSResult,
20     append_leaves,
21     can_be_split,
22     can_omit_invisible_parens,
23     is_line_short_enough,
24     line_to_string,
25 )
26 from black.mode import Feature, Mode, Preview
27 from black.nodes import (
28     ASSIGNMENTS,
29     BRACKETS,
30     CLOSING_BRACKETS,
31     OPENING_BRACKETS,
32     RARROW,
33     STANDALONE_COMMENT,
34     STATEMENT,
35     WHITESPACE,
36     Visitor,
37     ensure_visible,
38     is_arith_like,
39     is_async_stmt_or_funcdef,
40     is_atom_with_invisible_parens,
41     is_docstring,
42     is_empty_tuple,
43     is_lpar_token,
44     is_multiline_string,
45     is_name_token,
46     is_one_sequence_between,
47     is_one_tuple,
48     is_rpar_token,
49     is_stub_body,
50     is_stub_suite,
51     is_tuple_containing_walrus,
52     is_vararg,
53     is_walrus_assignment,
54     is_yield,
55     syms,
56     wrap_in_parentheses,
57 )
58 from black.numerics import normalize_numeric_literal
59 from black.strings import (
60     fix_docstring,
61     get_string_prefix,
62     normalize_string_prefix,
63     normalize_string_quotes,
64     normalize_unicode_escape_sequences,
65 )
66 from black.trans import (
67     CannotTransform,
68     StringMerger,
69     StringParenStripper,
70     StringParenWrapper,
71     StringSplitter,
72     Transformer,
73     hug_power_op,
74 )
75 from blib2to3.pgen2 import token
76 from blib2to3.pytree import Leaf, Node
77
78 # types
79 LeafID = int
80 LN = Union[Leaf, Node]
81
82
83 class CannotSplit(CannotTransform):
84     """A readable split that fits the allotted line length is impossible."""
85
86
87 # This isn't a dataclass because @dataclass + Generic breaks mypyc.
88 # See also https://github.com/mypyc/mypyc/issues/827.
89 class LineGenerator(Visitor[Line]):
90     """Generates reformatted Line objects.  Empty lines are not emitted.
91
92     Note: destroys the tree it's visiting by mutating prefixes of its leaves
93     in ways that will no longer stringify to valid Python code on the tree.
94     """
95
96     def __init__(self, mode: Mode, features: Collection[Feature]) -> None:
97         self.mode = mode
98         self.features = features
99         self.current_line: Line
100         self.__post_init__()
101
102     def line(self, indent: int = 0) -> Iterator[Line]:
103         """Generate a line.
104
105         If the line is empty, only emit if it makes sense.
106         If the line is too long, split it first and then generate.
107
108         If any lines were generated, set up a new current_line.
109         """
110         if not self.current_line:
111             self.current_line.depth += indent
112             return  # Line is empty, don't emit. Creating a new one unnecessary.
113
114         if (
115             Preview.improved_async_statements_handling in self.mode
116             and len(self.current_line.leaves) == 1
117             and is_async_stmt_or_funcdef(self.current_line.leaves[0])
118         ):
119             # Special case for async def/for/with statements. `visit_async_stmt`
120             # adds an `ASYNC` leaf then visits the child def/for/with statement
121             # nodes. Line yields from those nodes shouldn't treat the former
122             # `ASYNC` leaf as a complete line.
123             return
124
125         complete_line = self.current_line
126         self.current_line = Line(mode=self.mode, depth=complete_line.depth + indent)
127         yield complete_line
128
129     def visit_default(self, node: LN) -> Iterator[Line]:
130         """Default `visit_*()` implementation. Recurses to children of `node`."""
131         if isinstance(node, Leaf):
132             any_open_brackets = self.current_line.bracket_tracker.any_open_brackets()
133             for comment in generate_comments(node):
134                 if any_open_brackets:
135                     # any comment within brackets is subject to splitting
136                     self.current_line.append(comment)
137                 elif comment.type == token.COMMENT:
138                     # regular trailing comment
139                     self.current_line.append(comment)
140                     yield from self.line()
141
142                 else:
143                     # regular standalone comment
144                     yield from self.line()
145
146                     self.current_line.append(comment)
147                     yield from self.line()
148
149             normalize_prefix(node, inside_brackets=any_open_brackets)
150             if self.mode.string_normalization and node.type == token.STRING:
151                 node.value = normalize_string_prefix(node.value)
152                 node.value = normalize_string_quotes(node.value)
153             if node.type == token.NUMBER:
154                 normalize_numeric_literal(node)
155             if node.type not in WHITESPACE:
156                 self.current_line.append(node)
157         yield from super().visit_default(node)
158
159     def visit_test(self, node: Node) -> Iterator[Line]:
160         """Visit an `x if y else z` test"""
161
162         if Preview.parenthesize_conditional_expressions in self.mode:
163             already_parenthesized = (
164                 node.prev_sibling and node.prev_sibling.type == token.LPAR
165             )
166
167             if not already_parenthesized:
168                 lpar = Leaf(token.LPAR, "")
169                 rpar = Leaf(token.RPAR, "")
170                 node.insert_child(0, lpar)
171                 node.append_child(rpar)
172
173         yield from self.visit_default(node)
174
175     def visit_INDENT(self, node: Leaf) -> Iterator[Line]:
176         """Increase indentation level, maybe yield a line."""
177         # In blib2to3 INDENT never holds comments.
178         yield from self.line(+1)
179         yield from self.visit_default(node)
180
181     def visit_DEDENT(self, node: Leaf) -> Iterator[Line]:
182         """Decrease indentation level, maybe yield a line."""
183         # The current line might still wait for trailing comments.  At DEDENT time
184         # there won't be any (they would be prefixes on the preceding NEWLINE).
185         # Emit the line then.
186         yield from self.line()
187
188         # While DEDENT has no value, its prefix may contain standalone comments
189         # that belong to the current indentation level.  Get 'em.
190         yield from self.visit_default(node)
191
192         # Finally, emit the dedent.
193         yield from self.line(-1)
194
195     def visit_stmt(
196         self, node: Node, keywords: Set[str], parens: Set[str]
197     ) -> Iterator[Line]:
198         """Visit a statement.
199
200         This implementation is shared for `if`, `while`, `for`, `try`, `except`,
201         `def`, `with`, `class`, `assert`, and assignments.
202
203         The relevant Python language `keywords` for a given statement will be
204         NAME leaves within it. This methods puts those on a separate line.
205
206         `parens` holds a set of string leaf values immediately after which
207         invisible parens should be put.
208         """
209         normalize_invisible_parens(
210             node, parens_after=parens, mode=self.mode, features=self.features
211         )
212         for child in node.children:
213             if is_name_token(child) and child.value in keywords:
214                 yield from self.line()
215
216             yield from self.visit(child)
217
218     def visit_typeparams(self, node: Node) -> Iterator[Line]:
219         yield from self.visit_default(node)
220         node.children[0].prefix = ""
221
222     def visit_typevartuple(self, node: Node) -> Iterator[Line]:
223         yield from self.visit_default(node)
224         node.children[1].prefix = ""
225
226     def visit_paramspec(self, node: Node) -> Iterator[Line]:
227         yield from self.visit_default(node)
228         node.children[1].prefix = ""
229
230     def visit_dictsetmaker(self, node: Node) -> Iterator[Line]:
231         if Preview.wrap_long_dict_values_in_parens in self.mode:
232             for i, child in enumerate(node.children):
233                 if i == 0:
234                     continue
235                 if node.children[i - 1].type == token.COLON:
236                     if child.type == syms.atom and child.children[0].type == token.LPAR:
237                         if maybe_make_parens_invisible_in_atom(
238                             child,
239                             parent=node,
240                             remove_brackets_around_comma=False,
241                         ):
242                             wrap_in_parentheses(node, child, visible=False)
243                     else:
244                         wrap_in_parentheses(node, child, visible=False)
245         yield from self.visit_default(node)
246
247     def visit_funcdef(self, node: Node) -> Iterator[Line]:
248         """Visit function definition."""
249         yield from self.line()
250
251         # Remove redundant brackets around return type annotation.
252         is_return_annotation = False
253         for child in node.children:
254             if child.type == token.RARROW:
255                 is_return_annotation = True
256             elif is_return_annotation:
257                 if child.type == syms.atom and child.children[0].type == token.LPAR:
258                     if maybe_make_parens_invisible_in_atom(
259                         child,
260                         parent=node,
261                         remove_brackets_around_comma=False,
262                     ):
263                         wrap_in_parentheses(node, child, visible=False)
264                 else:
265                     wrap_in_parentheses(node, child, visible=False)
266                 is_return_annotation = False
267
268         for child in node.children:
269             yield from self.visit(child)
270
271     def visit_match_case(self, node: Node) -> Iterator[Line]:
272         """Visit either a match or case statement."""
273         normalize_invisible_parens(
274             node, parens_after=set(), mode=self.mode, features=self.features
275         )
276
277         yield from self.line()
278         for child in node.children:
279             yield from self.visit(child)
280
281     def visit_suite(self, node: Node) -> Iterator[Line]:
282         """Visit a suite."""
283         if self.mode.is_pyi and is_stub_suite(node):
284             yield from self.visit(node.children[2])
285         else:
286             yield from self.visit_default(node)
287
288     def visit_simple_stmt(self, node: Node) -> Iterator[Line]:
289         """Visit a statement without nested statements."""
290         prev_type: Optional[int] = None
291         for child in node.children:
292             if (prev_type is None or prev_type == token.SEMI) and is_arith_like(child):
293                 wrap_in_parentheses(node, child, visible=False)
294             prev_type = child.type
295
296         is_suite_like = node.parent and node.parent.type in STATEMENT
297         if is_suite_like:
298             if self.mode.is_pyi and is_stub_body(node):
299                 yield from self.visit_default(node)
300             else:
301                 yield from self.line(+1)
302                 yield from self.visit_default(node)
303                 yield from self.line(-1)
304
305         else:
306             if (
307                 not self.mode.is_pyi
308                 or not node.parent
309                 or not is_stub_suite(node.parent)
310             ):
311                 yield from self.line()
312             yield from self.visit_default(node)
313
314     def visit_async_stmt(self, node: Node) -> Iterator[Line]:
315         """Visit `async def`, `async for`, `async with`."""
316         yield from self.line()
317
318         children = iter(node.children)
319         for child in children:
320             yield from self.visit(child)
321
322             if child.type == token.ASYNC or child.type == STANDALONE_COMMENT:
323                 # STANDALONE_COMMENT happens when `# fmt: skip` is applied on the async
324                 # line.
325                 break
326
327         internal_stmt = next(children)
328         if Preview.improved_async_statements_handling in self.mode:
329             yield from self.visit(internal_stmt)
330         else:
331             for child in internal_stmt.children:
332                 yield from self.visit(child)
333
334     def visit_decorators(self, node: Node) -> Iterator[Line]:
335         """Visit decorators."""
336         for child in node.children:
337             yield from self.line()
338             yield from self.visit(child)
339
340     def visit_power(self, node: Node) -> Iterator[Line]:
341         for idx, leaf in enumerate(node.children[:-1]):
342             next_leaf = node.children[idx + 1]
343
344             if not isinstance(leaf, Leaf):
345                 continue
346
347             value = leaf.value.lower()
348             if (
349                 leaf.type == token.NUMBER
350                 and next_leaf.type == syms.trailer
351                 # Ensure that we are in an attribute trailer
352                 and next_leaf.children[0].type == token.DOT
353                 # It shouldn't wrap hexadecimal, binary and octal literals
354                 and not value.startswith(("0x", "0b", "0o"))
355                 # It shouldn't wrap complex literals
356                 and "j" not in value
357             ):
358                 wrap_in_parentheses(node, leaf)
359
360         remove_await_parens(node)
361
362         yield from self.visit_default(node)
363
364     def visit_SEMI(self, leaf: Leaf) -> Iterator[Line]:
365         """Remove a semicolon and put the other statement on a separate line."""
366         yield from self.line()
367
368     def visit_ENDMARKER(self, leaf: Leaf) -> Iterator[Line]:
369         """End of file. Process outstanding comments and end with a newline."""
370         yield from self.visit_default(leaf)
371         yield from self.line()
372
373     def visit_STANDALONE_COMMENT(self, leaf: Leaf) -> Iterator[Line]:
374         if not self.current_line.bracket_tracker.any_open_brackets():
375             yield from self.line()
376         yield from self.visit_default(leaf)
377
378     def visit_factor(self, node: Node) -> Iterator[Line]:
379         """Force parentheses between a unary op and a binary power:
380
381         -2 ** 8 -> -(2 ** 8)
382         """
383         _operator, operand = node.children
384         if (
385             operand.type == syms.power
386             and len(operand.children) == 3
387             and operand.children[1].type == token.DOUBLESTAR
388         ):
389             lpar = Leaf(token.LPAR, "(")
390             rpar = Leaf(token.RPAR, ")")
391             index = operand.remove() or 0
392             node.insert_child(index, Node(syms.atom, [lpar, operand, rpar]))
393         yield from self.visit_default(node)
394
395     def visit_STRING(self, leaf: Leaf) -> Iterator[Line]:
396         if Preview.hex_codes_in_unicode_sequences in self.mode:
397             normalize_unicode_escape_sequences(leaf)
398
399         if is_docstring(leaf) and "\\\n" not in leaf.value:
400             # We're ignoring docstrings with backslash newline escapes because changing
401             # indentation of those changes the AST representation of the code.
402             if self.mode.string_normalization:
403                 docstring = normalize_string_prefix(leaf.value)
404                 # visit_default() does handle string normalization for us, but
405                 # since this method acts differently depending on quote style (ex.
406                 # see padding logic below), there's a possibility for unstable
407                 # formatting as visit_default() is called *after*. To avoid a
408                 # situation where this function formats a docstring differently on
409                 # the second pass, normalize it early.
410                 docstring = normalize_string_quotes(docstring)
411             else:
412                 docstring = leaf.value
413             prefix = get_string_prefix(docstring)
414             docstring = docstring[len(prefix) :]  # Remove the prefix
415             quote_char = docstring[0]
416             # A natural way to remove the outer quotes is to do:
417             #   docstring = docstring.strip(quote_char)
418             # but that breaks on """""x""" (which is '""x').
419             # So we actually need to remove the first character and the next two
420             # characters but only if they are the same as the first.
421             quote_len = 1 if docstring[1] != quote_char else 3
422             docstring = docstring[quote_len:-quote_len]
423             docstring_started_empty = not docstring
424             indent = " " * 4 * self.current_line.depth
425
426             if is_multiline_string(leaf):
427                 docstring = fix_docstring(docstring, indent)
428             else:
429                 docstring = docstring.strip()
430
431             has_trailing_backslash = False
432             if docstring:
433                 # Add some padding if the docstring starts / ends with a quote mark.
434                 if docstring[0] == quote_char:
435                     docstring = " " + docstring
436                 if docstring[-1] == quote_char:
437                     docstring += " "
438                 if docstring[-1] == "\\":
439                     backslash_count = len(docstring) - len(docstring.rstrip("\\"))
440                     if backslash_count % 2:
441                         # Odd number of tailing backslashes, add some padding to
442                         # avoid escaping the closing string quote.
443                         docstring += " "
444                         has_trailing_backslash = True
445             elif not docstring_started_empty:
446                 docstring = " "
447
448             # We could enforce triple quotes at this point.
449             quote = quote_char * quote_len
450
451             # It's invalid to put closing single-character quotes on a new line.
452             if self.mode and quote_len == 3:
453                 # We need to find the length of the last line of the docstring
454                 # to find if we can add the closing quotes to the line without
455                 # exceeding the maximum line length.
456                 # If docstring is one line, we don't put the closing quotes on a
457                 # separate line because it looks ugly (#3320).
458                 lines = docstring.splitlines()
459                 last_line_length = len(lines[-1]) if docstring else 0
460
461                 # If adding closing quotes would cause the last line to exceed
462                 # the maximum line length then put a line break before the
463                 # closing quotes
464                 if (
465                     len(lines) > 1
466                     and last_line_length + quote_len > self.mode.line_length
467                     and len(indent) + quote_len <= self.mode.line_length
468                     and not has_trailing_backslash
469                 ):
470                     leaf.value = prefix + quote + docstring + "\n" + indent + quote
471                 else:
472                     leaf.value = prefix + quote + docstring + quote
473             else:
474                 leaf.value = prefix + quote + docstring + quote
475
476         yield from self.visit_default(leaf)
477
478     def __post_init__(self) -> None:
479         """You are in a twisty little maze of passages."""
480         self.current_line = Line(mode=self.mode)
481
482         v = self.visit_stmt
483         Ø: Set[str] = set()
484         self.visit_assert_stmt = partial(v, keywords={"assert"}, parens={"assert", ","})
485         self.visit_if_stmt = partial(
486             v, keywords={"if", "else", "elif"}, parens={"if", "elif"}
487         )
488         self.visit_while_stmt = partial(v, keywords={"while", "else"}, parens={"while"})
489         self.visit_for_stmt = partial(v, keywords={"for", "else"}, parens={"for", "in"})
490         self.visit_try_stmt = partial(
491             v, keywords={"try", "except", "else", "finally"}, parens=Ø
492         )
493         self.visit_except_clause = partial(v, keywords={"except"}, parens={"except"})
494         self.visit_with_stmt = partial(v, keywords={"with"}, parens={"with"})
495         self.visit_classdef = partial(v, keywords={"class"}, parens=Ø)
496         self.visit_expr_stmt = partial(v, keywords=Ø, parens=ASSIGNMENTS)
497         self.visit_return_stmt = partial(v, keywords={"return"}, parens={"return"})
498         self.visit_import_from = partial(v, keywords=Ø, parens={"import"})
499         self.visit_del_stmt = partial(v, keywords=Ø, parens={"del"})
500         self.visit_async_funcdef = self.visit_async_stmt
501         self.visit_decorated = self.visit_decorators
502
503         # PEP 634
504         self.visit_match_stmt = self.visit_match_case
505         self.visit_case_block = self.visit_match_case
506
507
508 def transform_line(
509     line: Line, mode: Mode, features: Collection[Feature] = ()
510 ) -> Iterator[Line]:
511     """Transform a `line`, potentially splitting it into many lines.
512
513     They should fit in the allotted `line_length` but might not be able to.
514
515     `features` are syntactical features that may be used in the output.
516     """
517     if line.is_comment:
518         yield line
519         return
520
521     line_str = line_to_string(line)
522
523     ll = mode.line_length
524     sn = mode.string_normalization
525     string_merge = StringMerger(ll, sn)
526     string_paren_strip = StringParenStripper(ll, sn)
527     string_split = StringSplitter(ll, sn)
528     string_paren_wrap = StringParenWrapper(ll, sn)
529
530     transformers: List[Transformer]
531     if (
532         not line.contains_uncollapsable_type_comments()
533         and not line.should_split_rhs
534         and not line.magic_trailing_comma
535         and (
536             is_line_short_enough(line, mode=mode, line_str=line_str)
537             or line.contains_unsplittable_type_ignore()
538         )
539         and not (line.inside_brackets and line.contains_standalone_comments())
540     ):
541         # Only apply basic string preprocessing, since lines shouldn't be split here.
542         if Preview.string_processing in mode:
543             transformers = [string_merge, string_paren_strip]
544         else:
545             transformers = []
546     elif line.is_def:
547         transformers = [left_hand_split]
548     else:
549
550         def _rhs(
551             self: object, line: Line, features: Collection[Feature], mode: Mode
552         ) -> Iterator[Line]:
553             """Wraps calls to `right_hand_split`.
554
555             The calls increasingly `omit` right-hand trailers (bracket pairs with
556             content), meaning the trailers get glued together to split on another
557             bracket pair instead.
558             """
559             for omit in generate_trailers_to_omit(line, mode.line_length):
560                 lines = list(right_hand_split(line, mode, features, omit=omit))
561                 # Note: this check is only able to figure out if the first line of the
562                 # *current* transformation fits in the line length.  This is true only
563                 # for simple cases.  All others require running more transforms via
564                 # `transform_line()`.  This check doesn't know if those would succeed.
565                 if is_line_short_enough(lines[0], mode=mode):
566                     yield from lines
567                     return
568
569             # All splits failed, best effort split with no omits.
570             # This mostly happens to multiline strings that are by definition
571             # reported as not fitting a single line, as well as lines that contain
572             # trailing commas (those have to be exploded).
573             yield from right_hand_split(line, mode, features=features)
574
575         # HACK: nested functions (like _rhs) compiled by mypyc don't retain their
576         # __name__ attribute which is needed in `run_transformer` further down.
577         # Unfortunately a nested class breaks mypyc too. So a class must be created
578         # via type ... https://github.com/mypyc/mypyc/issues/884
579         rhs = type("rhs", (), {"__call__": _rhs})()
580
581         if Preview.string_processing in mode:
582             if line.inside_brackets:
583                 transformers = [
584                     string_merge,
585                     string_paren_strip,
586                     string_split,
587                     delimiter_split,
588                     standalone_comment_split,
589                     string_paren_wrap,
590                     rhs,
591                 ]
592             else:
593                 transformers = [
594                     string_merge,
595                     string_paren_strip,
596                     string_split,
597                     string_paren_wrap,
598                     rhs,
599                 ]
600         else:
601             if line.inside_brackets:
602                 transformers = [delimiter_split, standalone_comment_split, rhs]
603             else:
604                 transformers = [rhs]
605     # It's always safe to attempt hugging of power operations and pretty much every line
606     # could match.
607     transformers.append(hug_power_op)
608
609     for transform in transformers:
610         # We are accumulating lines in `result` because we might want to abort
611         # mission and return the original line in the end, or attempt a different
612         # split altogether.
613         try:
614             result = run_transformer(line, transform, mode, features, line_str=line_str)
615         except CannotTransform:
616             continue
617         else:
618             yield from result
619             break
620
621     else:
622         yield line
623
624
625 class _BracketSplitComponent(Enum):
626     head = auto()
627     body = auto()
628     tail = auto()
629
630
631 def left_hand_split(
632     line: Line, _features: Collection[Feature], mode: Mode
633 ) -> Iterator[Line]:
634     """Split line into many lines, starting with the first matching bracket pair.
635
636     Note: this usually looks weird, only use this for function definitions.
637     Prefer RHS otherwise.  This is why this function is not symmetrical with
638     :func:`right_hand_split` which also handles optional parentheses.
639     """
640     tail_leaves: List[Leaf] = []
641     body_leaves: List[Leaf] = []
642     head_leaves: List[Leaf] = []
643     current_leaves = head_leaves
644     matching_bracket: Optional[Leaf] = None
645     for leaf in line.leaves:
646         if (
647             current_leaves is body_leaves
648             and leaf.type in CLOSING_BRACKETS
649             and leaf.opening_bracket is matching_bracket
650             and isinstance(matching_bracket, Leaf)
651         ):
652             ensure_visible(leaf)
653             ensure_visible(matching_bracket)
654             current_leaves = tail_leaves if body_leaves else head_leaves
655         current_leaves.append(leaf)
656         if current_leaves is head_leaves:
657             if leaf.type in OPENING_BRACKETS:
658                 matching_bracket = leaf
659                 current_leaves = body_leaves
660     if not matching_bracket:
661         raise CannotSplit("No brackets found")
662
663     head = bracket_split_build_line(
664         head_leaves, line, matching_bracket, component=_BracketSplitComponent.head
665     )
666     body = bracket_split_build_line(
667         body_leaves, line, matching_bracket, component=_BracketSplitComponent.body
668     )
669     tail = bracket_split_build_line(
670         tail_leaves, line, matching_bracket, component=_BracketSplitComponent.tail
671     )
672     bracket_split_succeeded_or_raise(head, body, tail)
673     for result in (head, body, tail):
674         if result:
675             yield result
676
677
678 def right_hand_split(
679     line: Line,
680     mode: Mode,
681     features: Collection[Feature] = (),
682     omit: Collection[LeafID] = (),
683 ) -> Iterator[Line]:
684     """Split line into many lines, starting with the last matching bracket pair.
685
686     If the split was by optional parentheses, attempt splitting without them, too.
687     `omit` is a collection of closing bracket IDs that shouldn't be considered for
688     this split.
689
690     Note: running this function modifies `bracket_depth` on the leaves of `line`.
691     """
692     rhs_result = _first_right_hand_split(line, omit=omit)
693     yield from _maybe_split_omitting_optional_parens(
694         rhs_result, line, mode, features=features, omit=omit
695     )
696
697
698 def _first_right_hand_split(
699     line: Line,
700     omit: Collection[LeafID] = (),
701 ) -> RHSResult:
702     """Split the line into head, body, tail starting with the last bracket pair.
703
704     Note: this function should not have side effects. It's relied upon by
705     _maybe_split_omitting_optional_parens to get an opinion whether to prefer
706     splitting on the right side of an assignment statement.
707     """
708     tail_leaves: List[Leaf] = []
709     body_leaves: List[Leaf] = []
710     head_leaves: List[Leaf] = []
711     current_leaves = tail_leaves
712     opening_bracket: Optional[Leaf] = None
713     closing_bracket: Optional[Leaf] = None
714     for leaf in reversed(line.leaves):
715         if current_leaves is body_leaves:
716             if leaf is opening_bracket:
717                 current_leaves = head_leaves if body_leaves else tail_leaves
718         current_leaves.append(leaf)
719         if current_leaves is tail_leaves:
720             if leaf.type in CLOSING_BRACKETS and id(leaf) not in omit:
721                 opening_bracket = leaf.opening_bracket
722                 closing_bracket = leaf
723                 current_leaves = body_leaves
724     if not (opening_bracket and closing_bracket and head_leaves):
725         # If there is no opening or closing_bracket that means the split failed and
726         # all content is in the tail.  Otherwise, if `head_leaves` are empty, it means
727         # the matching `opening_bracket` wasn't available on `line` anymore.
728         raise CannotSplit("No brackets found")
729
730     tail_leaves.reverse()
731     body_leaves.reverse()
732     head_leaves.reverse()
733     head = bracket_split_build_line(
734         head_leaves, line, opening_bracket, component=_BracketSplitComponent.head
735     )
736     body = bracket_split_build_line(
737         body_leaves, line, opening_bracket, component=_BracketSplitComponent.body
738     )
739     tail = bracket_split_build_line(
740         tail_leaves, line, opening_bracket, component=_BracketSplitComponent.tail
741     )
742     bracket_split_succeeded_or_raise(head, body, tail)
743     return RHSResult(head, body, tail, opening_bracket, closing_bracket)
744
745
746 def _maybe_split_omitting_optional_parens(
747     rhs: RHSResult,
748     line: Line,
749     mode: Mode,
750     features: Collection[Feature] = (),
751     omit: Collection[LeafID] = (),
752 ) -> Iterator[Line]:
753     if (
754         Feature.FORCE_OPTIONAL_PARENTHESES not in features
755         # the opening bracket is an optional paren
756         and rhs.opening_bracket.type == token.LPAR
757         and not rhs.opening_bracket.value
758         # the closing bracket is an optional paren
759         and rhs.closing_bracket.type == token.RPAR
760         and not rhs.closing_bracket.value
761         # it's not an import (optional parens are the only thing we can split on
762         # in this case; attempting a split without them is a waste of time)
763         and not line.is_import
764         # there are no standalone comments in the body
765         and not rhs.body.contains_standalone_comments(0)
766         # and we can actually remove the parens
767         and can_omit_invisible_parens(rhs, mode.line_length)
768     ):
769         omit = {id(rhs.closing_bracket), *omit}
770         try:
771             # The RHSResult Omitting Optional Parens.
772             rhs_oop = _first_right_hand_split(line, omit=omit)
773             if not (
774                 Preview.prefer_splitting_right_hand_side_of_assignments in line.mode
775                 # the split is right after `=`
776                 and len(rhs.head.leaves) >= 2
777                 and rhs.head.leaves[-2].type == token.EQUAL
778                 # the left side of assignment contains brackets
779                 and any(leaf.type in BRACKETS for leaf in rhs.head.leaves[:-1])
780                 # the left side of assignment is short enough (the -1 is for the ending
781                 # optional paren)
782                 and is_line_short_enough(
783                     rhs.head, mode=replace(mode, line_length=mode.line_length - 1)
784                 )
785                 # the left side of assignment won't explode further because of magic
786                 # trailing comma
787                 and rhs.head.magic_trailing_comma is None
788                 # the split by omitting optional parens isn't preferred by some other
789                 # reason
790                 and not _prefer_split_rhs_oop(rhs_oop, mode)
791             ):
792                 yield from _maybe_split_omitting_optional_parens(
793                     rhs_oop, line, mode, features=features, omit=omit
794                 )
795                 return
796
797         except CannotSplit as e:
798             if not (
799                 can_be_split(rhs.body) or is_line_short_enough(rhs.body, mode=mode)
800             ):
801                 raise CannotSplit(
802                     "Splitting failed, body is still too long and can't be split."
803                 ) from e
804
805             elif (
806                 rhs.head.contains_multiline_strings()
807                 or rhs.tail.contains_multiline_strings()
808             ):
809                 raise CannotSplit(
810                     "The current optional pair of parentheses is bound to fail to"
811                     " satisfy the splitting algorithm because the head or the tail"
812                     " contains multiline strings which by definition never fit one"
813                     " line."
814                 ) from e
815
816     ensure_visible(rhs.opening_bracket)
817     ensure_visible(rhs.closing_bracket)
818     for result in (rhs.head, rhs.body, rhs.tail):
819         if result:
820             yield result
821
822
823 def _prefer_split_rhs_oop(rhs_oop: RHSResult, mode: Mode) -> bool:
824     """
825     Returns whether we should prefer the result from a split omitting optional parens.
826     """
827     has_closing_bracket_after_assign = False
828     for leaf in reversed(rhs_oop.head.leaves):
829         if leaf.type == token.EQUAL:
830             break
831         if leaf.type in CLOSING_BRACKETS:
832             has_closing_bracket_after_assign = True
833             break
834     return (
835         # contains matching brackets after the `=` (done by checking there is a
836         # closing bracket)
837         has_closing_bracket_after_assign
838         or (
839             # the split is actually from inside the optional parens (done by checking
840             # the first line still contains the `=`)
841             any(leaf.type == token.EQUAL for leaf in rhs_oop.head.leaves)
842             # the first line is short enough
843             and is_line_short_enough(rhs_oop.head, mode=mode)
844         )
845         # contains unsplittable type ignore
846         or rhs_oop.head.contains_unsplittable_type_ignore()
847         or rhs_oop.body.contains_unsplittable_type_ignore()
848         or rhs_oop.tail.contains_unsplittable_type_ignore()
849     )
850
851
852 def bracket_split_succeeded_or_raise(head: Line, body: Line, tail: Line) -> None:
853     """Raise :exc:`CannotSplit` if the last left- or right-hand split failed.
854
855     Do nothing otherwise.
856
857     A left- or right-hand split is based on a pair of brackets. Content before
858     (and including) the opening bracket is left on one line, content inside the
859     brackets is put on a separate line, and finally content starting with and
860     following the closing bracket is put on a separate line.
861
862     Those are called `head`, `body`, and `tail`, respectively. If the split
863     produced the same line (all content in `head`) or ended up with an empty `body`
864     and the `tail` is just the closing bracket, then it's considered failed.
865     """
866     tail_len = len(str(tail).strip())
867     if not body:
868         if tail_len == 0:
869             raise CannotSplit("Splitting brackets produced the same line")
870
871         elif tail_len < 3:
872             raise CannotSplit(
873                 f"Splitting brackets on an empty body to save {tail_len} characters is"
874                 " not worth it"
875             )
876
877
878 def bracket_split_build_line(
879     leaves: List[Leaf],
880     original: Line,
881     opening_bracket: Leaf,
882     *,
883     component: _BracketSplitComponent,
884 ) -> Line:
885     """Return a new line with given `leaves` and respective comments from `original`.
886
887     If it's the head component, brackets will be tracked so trailing commas are
888     respected.
889
890     If it's the body component, the result line is one-indented inside brackets and as
891     such has its first leaf's prefix normalized and a trailing comma added when
892     expected.
893     """
894     result = Line(mode=original.mode, depth=original.depth)
895     if component is _BracketSplitComponent.body:
896         result.inside_brackets = True
897         result.depth += 1
898         if leaves:
899             # Since body is a new indent level, remove spurious leading whitespace.
900             normalize_prefix(leaves[0], inside_brackets=True)
901             # Ensure a trailing comma for imports and standalone function arguments, but
902             # be careful not to add one after any comments or within type annotations.
903             no_commas = (
904                 original.is_def
905                 and opening_bracket.value == "("
906                 and not any(leaf.type == token.COMMA for leaf in leaves)
907                 # In particular, don't add one within a parenthesized return annotation.
908                 # Unfortunately the indicator we're in a return annotation (RARROW) may
909                 # be defined directly in the parent node, the parent of the parent ...
910                 # and so on depending on how complex the return annotation is.
911                 # This isn't perfect and there's some false negatives but they are in
912                 # contexts were a comma is actually fine.
913                 and not any(
914                     node.prev_sibling.type == RARROW
915                     for node in (
916                         leaves[0].parent,
917                         getattr(leaves[0].parent, "parent", None),
918                     )
919                     if isinstance(node, Node) and isinstance(node.prev_sibling, Leaf)
920                 )
921                 # Except the false negatives above for PEP 604 unions where we
922                 # can't add the comma.
923                 and not (
924                     leaves[0].parent
925                     and leaves[0].parent.next_sibling
926                     and leaves[0].parent.next_sibling.type == token.VBAR
927                 )
928             )
929
930             if original.is_import or no_commas:
931                 for i in range(len(leaves) - 1, -1, -1):
932                     if leaves[i].type == STANDALONE_COMMENT:
933                         continue
934
935                     if leaves[i].type != token.COMMA:
936                         new_comma = Leaf(token.COMMA, ",")
937                         leaves.insert(i + 1, new_comma)
938                     break
939
940     leaves_to_track: Set[LeafID] = set()
941     if component is _BracketSplitComponent.head:
942         leaves_to_track = get_leaves_inside_matching_brackets(leaves)
943     # Populate the line
944     for leaf in leaves:
945         result.append(
946             leaf,
947             preformatted=True,
948             track_bracket=id(leaf) in leaves_to_track,
949         )
950         for comment_after in original.comments_after(leaf):
951             result.append(comment_after, preformatted=True)
952     if component is _BracketSplitComponent.body and should_split_line(
953         result, opening_bracket
954     ):
955         result.should_split_rhs = True
956     return result
957
958
959 def dont_increase_indentation(split_func: Transformer) -> Transformer:
960     """Normalize prefix of the first leaf in every line returned by `split_func`.
961
962     This is a decorator over relevant split functions.
963     """
964
965     @wraps(split_func)
966     def split_wrapper(
967         line: Line, features: Collection[Feature], mode: Mode
968     ) -> Iterator[Line]:
969         for split_line in split_func(line, features, mode):
970             normalize_prefix(split_line.leaves[0], inside_brackets=True)
971             yield split_line
972
973     return split_wrapper
974
975
976 def _get_last_non_comment_leaf(line: Line) -> Optional[int]:
977     for leaf_idx in range(len(line.leaves) - 1, 0, -1):
978         if line.leaves[leaf_idx].type != STANDALONE_COMMENT:
979             return leaf_idx
980     return None
981
982
983 def _safe_add_trailing_comma(safe: bool, delimiter_priority: int, line: Line) -> Line:
984     if (
985         safe
986         and delimiter_priority == COMMA_PRIORITY
987         and line.leaves[-1].type != token.COMMA
988         and line.leaves[-1].type != STANDALONE_COMMENT
989     ):
990         new_comma = Leaf(token.COMMA, ",")
991         line.append(new_comma)
992     return line
993
994
995 @dont_increase_indentation
996 def delimiter_split(
997     line: Line, features: Collection[Feature], mode: Mode
998 ) -> Iterator[Line]:
999     """Split according to delimiters of the highest priority.
1000
1001     If the appropriate Features are given, the split will add trailing commas
1002     also in function signatures and calls that contain `*` and `**`.
1003     """
1004     try:
1005         last_leaf = line.leaves[-1]
1006     except IndexError:
1007         raise CannotSplit("Line empty") from None
1008
1009     bt = line.bracket_tracker
1010     try:
1011         delimiter_priority = bt.max_delimiter_priority(exclude={id(last_leaf)})
1012     except ValueError:
1013         raise CannotSplit("No delimiters found") from None
1014
1015     if delimiter_priority == DOT_PRIORITY:
1016         if bt.delimiter_count_with_priority(delimiter_priority) == 1:
1017             raise CannotSplit("Splitting a single attribute from its owner looks wrong")
1018
1019     current_line = Line(
1020         mode=line.mode, depth=line.depth, inside_brackets=line.inside_brackets
1021     )
1022     lowest_depth = sys.maxsize
1023     trailing_comma_safe = True
1024
1025     def append_to_line(leaf: Leaf) -> Iterator[Line]:
1026         """Append `leaf` to current line or to new line if appending impossible."""
1027         nonlocal current_line
1028         try:
1029             current_line.append_safe(leaf, preformatted=True)
1030         except ValueError:
1031             yield current_line
1032
1033             current_line = Line(
1034                 mode=line.mode, depth=line.depth, inside_brackets=line.inside_brackets
1035             )
1036             current_line.append(leaf)
1037
1038     last_non_comment_leaf = _get_last_non_comment_leaf(line)
1039     for leaf_idx, leaf in enumerate(line.leaves):
1040         yield from append_to_line(leaf)
1041
1042         for comment_after in line.comments_after(leaf):
1043             yield from append_to_line(comment_after)
1044
1045         lowest_depth = min(lowest_depth, leaf.bracket_depth)
1046         if leaf.bracket_depth == lowest_depth:
1047             if is_vararg(leaf, within={syms.typedargslist}):
1048                 trailing_comma_safe = (
1049                     trailing_comma_safe and Feature.TRAILING_COMMA_IN_DEF in features
1050                 )
1051             elif is_vararg(leaf, within={syms.arglist, syms.argument}):
1052                 trailing_comma_safe = (
1053                     trailing_comma_safe and Feature.TRAILING_COMMA_IN_CALL in features
1054                 )
1055
1056         if (
1057             Preview.add_trailing_comma_consistently in mode
1058             and last_leaf.type == STANDALONE_COMMENT
1059             and leaf_idx == last_non_comment_leaf
1060         ):
1061             current_line = _safe_add_trailing_comma(
1062                 trailing_comma_safe, delimiter_priority, current_line
1063             )
1064
1065         leaf_priority = bt.delimiters.get(id(leaf))
1066         if leaf_priority == delimiter_priority:
1067             yield current_line
1068
1069             current_line = Line(
1070                 mode=line.mode, depth=line.depth, inside_brackets=line.inside_brackets
1071             )
1072     if current_line:
1073         current_line = _safe_add_trailing_comma(
1074             trailing_comma_safe, delimiter_priority, current_line
1075         )
1076         yield current_line
1077
1078
1079 @dont_increase_indentation
1080 def standalone_comment_split(
1081     line: Line, features: Collection[Feature], mode: Mode
1082 ) -> Iterator[Line]:
1083     """Split standalone comments from the rest of the line."""
1084     if not line.contains_standalone_comments(0):
1085         raise CannotSplit("Line does not have any standalone comments")
1086
1087     current_line = Line(
1088         mode=line.mode, depth=line.depth, inside_brackets=line.inside_brackets
1089     )
1090
1091     def append_to_line(leaf: Leaf) -> Iterator[Line]:
1092         """Append `leaf` to current line or to new line if appending impossible."""
1093         nonlocal current_line
1094         try:
1095             current_line.append_safe(leaf, preformatted=True)
1096         except ValueError:
1097             yield current_line
1098
1099             current_line = Line(
1100                 line.mode, depth=line.depth, inside_brackets=line.inside_brackets
1101             )
1102             current_line.append(leaf)
1103
1104     for leaf in line.leaves:
1105         yield from append_to_line(leaf)
1106
1107         for comment_after in line.comments_after(leaf):
1108             yield from append_to_line(comment_after)
1109
1110     if current_line:
1111         yield current_line
1112
1113
1114 def normalize_prefix(leaf: Leaf, *, inside_brackets: bool) -> None:
1115     """Leave existing extra newlines if not `inside_brackets`. Remove everything
1116     else.
1117
1118     Note: don't use backslashes for formatting or you'll lose your voting rights.
1119     """
1120     if not inside_brackets:
1121         spl = leaf.prefix.split("#")
1122         if "\\" not in spl[0]:
1123             nl_count = spl[-1].count("\n")
1124             if len(spl) > 1:
1125                 nl_count -= 1
1126             leaf.prefix = "\n" * nl_count
1127             return
1128
1129     leaf.prefix = ""
1130
1131
1132 def normalize_invisible_parens(
1133     node: Node, parens_after: Set[str], *, mode: Mode, features: Collection[Feature]
1134 ) -> None:
1135     """Make existing optional parentheses invisible or create new ones.
1136
1137     `parens_after` is a set of string leaf values immediately after which parens
1138     should be put.
1139
1140     Standardizes on visible parentheses for single-element tuples, and keeps
1141     existing visible parentheses for other tuples and generator expressions.
1142     """
1143     for pc in list_comments(node.prefix, is_endmarker=False):
1144         if pc.value in FMT_OFF:
1145             # This `node` has a prefix with `# fmt: off`, don't mess with parens.
1146             return
1147
1148     # The multiple context managers grammar has a different pattern, thus this is
1149     # separate from the for-loop below. This possibly wraps them in invisible parens,
1150     # and later will be removed in remove_with_parens when needed.
1151     if node.type == syms.with_stmt:
1152         _maybe_wrap_cms_in_parens(node, mode, features)
1153
1154     check_lpar = False
1155     for index, child in enumerate(list(node.children)):
1156         # Fixes a bug where invisible parens are not properly stripped from
1157         # assignment statements that contain type annotations.
1158         if isinstance(child, Node) and child.type == syms.annassign:
1159             normalize_invisible_parens(
1160                 child, parens_after=parens_after, mode=mode, features=features
1161             )
1162
1163         # Add parentheses around long tuple unpacking in assignments.
1164         if (
1165             index == 0
1166             and isinstance(child, Node)
1167             and child.type == syms.testlist_star_expr
1168         ):
1169             check_lpar = True
1170
1171         if check_lpar:
1172             if (
1173                 child.type == syms.atom
1174                 and node.type == syms.for_stmt
1175                 and isinstance(child.prev_sibling, Leaf)
1176                 and child.prev_sibling.type == token.NAME
1177                 and child.prev_sibling.value == "for"
1178             ):
1179                 if maybe_make_parens_invisible_in_atom(
1180                     child,
1181                     parent=node,
1182                     remove_brackets_around_comma=True,
1183                 ):
1184                     wrap_in_parentheses(node, child, visible=False)
1185             elif isinstance(child, Node) and node.type == syms.with_stmt:
1186                 remove_with_parens(child, node)
1187             elif child.type == syms.atom:
1188                 if maybe_make_parens_invisible_in_atom(
1189                     child,
1190                     parent=node,
1191                 ):
1192                     wrap_in_parentheses(node, child, visible=False)
1193             elif is_one_tuple(child):
1194                 wrap_in_parentheses(node, child, visible=True)
1195             elif node.type == syms.import_from:
1196                 _normalize_import_from(node, child, index)
1197                 break
1198             elif (
1199                 index == 1
1200                 and child.type == token.STAR
1201                 and node.type == syms.except_clause
1202             ):
1203                 # In except* (PEP 654), the star is actually part of
1204                 # of the keyword. So we need to skip the insertion of
1205                 # invisible parentheses to work more precisely.
1206                 continue
1207
1208             elif not (isinstance(child, Leaf) and is_multiline_string(child)):
1209                 wrap_in_parentheses(node, child, visible=False)
1210
1211         comma_check = child.type == token.COMMA
1212
1213         check_lpar = isinstance(child, Leaf) and (
1214             child.value in parens_after or comma_check
1215         )
1216
1217
1218 def _normalize_import_from(parent: Node, child: LN, index: int) -> None:
1219     # "import from" nodes store parentheses directly as part of
1220     # the statement
1221     if is_lpar_token(child):
1222         assert is_rpar_token(parent.children[-1])
1223         # make parentheses invisible
1224         child.value = ""
1225         parent.children[-1].value = ""
1226     elif child.type != token.STAR:
1227         # insert invisible parentheses
1228         parent.insert_child(index, Leaf(token.LPAR, ""))
1229         parent.append_child(Leaf(token.RPAR, ""))
1230
1231
1232 def remove_await_parens(node: Node) -> None:
1233     if node.children[0].type == token.AWAIT and len(node.children) > 1:
1234         if (
1235             node.children[1].type == syms.atom
1236             and node.children[1].children[0].type == token.LPAR
1237         ):
1238             if maybe_make_parens_invisible_in_atom(
1239                 node.children[1],
1240                 parent=node,
1241                 remove_brackets_around_comma=True,
1242             ):
1243                 wrap_in_parentheses(node, node.children[1], visible=False)
1244
1245             # Since await is an expression we shouldn't remove
1246             # brackets in cases where this would change
1247             # the AST due to operator precedence.
1248             # Therefore we only aim to remove brackets around
1249             # power nodes that aren't also await expressions themselves.
1250             # https://peps.python.org/pep-0492/#updated-operator-precedence-table
1251             # N.B. We've still removed any redundant nested brackets though :)
1252             opening_bracket = cast(Leaf, node.children[1].children[0])
1253             closing_bracket = cast(Leaf, node.children[1].children[-1])
1254             bracket_contents = node.children[1].children[1]
1255             if isinstance(bracket_contents, Node):
1256                 if bracket_contents.type != syms.power:
1257                     ensure_visible(opening_bracket)
1258                     ensure_visible(closing_bracket)
1259                 elif (
1260                     bracket_contents.type == syms.power
1261                     and bracket_contents.children[0].type == token.AWAIT
1262                 ):
1263                     ensure_visible(opening_bracket)
1264                     ensure_visible(closing_bracket)
1265                     # If we are in a nested await then recurse down.
1266                     remove_await_parens(bracket_contents)
1267
1268
1269 def _maybe_wrap_cms_in_parens(
1270     node: Node, mode: Mode, features: Collection[Feature]
1271 ) -> None:
1272     """When enabled and safe, wrap the multiple context managers in invisible parens.
1273
1274     It is only safe when `features` contain Feature.PARENTHESIZED_CONTEXT_MANAGERS.
1275     """
1276     if (
1277         Feature.PARENTHESIZED_CONTEXT_MANAGERS not in features
1278         or Preview.wrap_multiple_context_managers_in_parens not in mode
1279         or len(node.children) <= 2
1280         # If it's an atom, it's already wrapped in parens.
1281         or node.children[1].type == syms.atom
1282     ):
1283         return
1284     colon_index: Optional[int] = None
1285     for i in range(2, len(node.children)):
1286         if node.children[i].type == token.COLON:
1287             colon_index = i
1288             break
1289     if colon_index is not None:
1290         lpar = Leaf(token.LPAR, "")
1291         rpar = Leaf(token.RPAR, "")
1292         context_managers = node.children[1:colon_index]
1293         for child in context_managers:
1294             child.remove()
1295         # After wrapping, the with_stmt will look like this:
1296         #   with_stmt
1297         #     NAME 'with'
1298         #     atom
1299         #       LPAR ''
1300         #       testlist_gexp
1301         #         ... <-- context_managers
1302         #       /testlist_gexp
1303         #       RPAR ''
1304         #     /atom
1305         #     COLON ':'
1306         new_child = Node(
1307             syms.atom, [lpar, Node(syms.testlist_gexp, context_managers), rpar]
1308         )
1309         node.insert_child(1, new_child)
1310
1311
1312 def remove_with_parens(node: Node, parent: Node) -> None:
1313     """Recursively hide optional parens in `with` statements."""
1314     # Removing all unnecessary parentheses in with statements in one pass is a tad
1315     # complex as different variations of bracketed statements result in pretty
1316     # different parse trees:
1317     #
1318     # with (open("file")) as f:                       # this is an asexpr_test
1319     #     ...
1320     #
1321     # with (open("file") as f):                       # this is an atom containing an
1322     #     ...                                         # asexpr_test
1323     #
1324     # with (open("file")) as f, (open("file")) as f:  # this is asexpr_test, COMMA,
1325     #     ...                                         # asexpr_test
1326     #
1327     # with (open("file") as f, open("file") as f):    # an atom containing a
1328     #     ...                                         # testlist_gexp which then
1329     #                                                 # contains multiple asexpr_test(s)
1330     if node.type == syms.atom:
1331         if maybe_make_parens_invisible_in_atom(
1332             node,
1333             parent=parent,
1334             remove_brackets_around_comma=True,
1335         ):
1336             wrap_in_parentheses(parent, node, visible=False)
1337         if isinstance(node.children[1], Node):
1338             remove_with_parens(node.children[1], node)
1339     elif node.type == syms.testlist_gexp:
1340         for child in node.children:
1341             if isinstance(child, Node):
1342                 remove_with_parens(child, node)
1343     elif node.type == syms.asexpr_test and not any(
1344         leaf.type == token.COLONEQUAL for leaf in node.leaves()
1345     ):
1346         if maybe_make_parens_invisible_in_atom(
1347             node.children[0],
1348             parent=node,
1349             remove_brackets_around_comma=True,
1350         ):
1351             wrap_in_parentheses(node, node.children[0], visible=False)
1352
1353
1354 def maybe_make_parens_invisible_in_atom(
1355     node: LN,
1356     parent: LN,
1357     remove_brackets_around_comma: bool = False,
1358 ) -> bool:
1359     """If it's safe, make the parens in the atom `node` invisible, recursively.
1360     Additionally, remove repeated, adjacent invisible parens from the atom `node`
1361     as they are redundant.
1362
1363     Returns whether the node should itself be wrapped in invisible parentheses.
1364     """
1365     if (
1366         node.type != syms.atom
1367         or is_empty_tuple(node)
1368         or is_one_tuple(node)
1369         or (is_yield(node) and parent.type != syms.expr_stmt)
1370         or (
1371             # This condition tries to prevent removing non-optional brackets
1372             # around a tuple, however, can be a bit overzealous so we provide
1373             # and option to skip this check for `for` and `with` statements.
1374             not remove_brackets_around_comma
1375             and max_delimiter_priority_in_atom(node) >= COMMA_PRIORITY
1376         )
1377         or is_tuple_containing_walrus(node)
1378     ):
1379         return False
1380
1381     if is_walrus_assignment(node):
1382         if parent.type in [
1383             syms.annassign,
1384             syms.expr_stmt,
1385             syms.assert_stmt,
1386             syms.return_stmt,
1387             syms.except_clause,
1388             syms.funcdef,
1389             syms.with_stmt,
1390             # these ones aren't useful to end users, but they do please fuzzers
1391             syms.for_stmt,
1392             syms.del_stmt,
1393             syms.for_stmt,
1394         ]:
1395             return False
1396
1397     first = node.children[0]
1398     last = node.children[-1]
1399     if is_lpar_token(first) and is_rpar_token(last):
1400         middle = node.children[1]
1401         # make parentheses invisible
1402         first.value = ""
1403         last.value = ""
1404         maybe_make_parens_invisible_in_atom(
1405             middle,
1406             parent=parent,
1407             remove_brackets_around_comma=remove_brackets_around_comma,
1408         )
1409
1410         if is_atom_with_invisible_parens(middle):
1411             # Strip the invisible parens from `middle` by replacing
1412             # it with the child in-between the invisible parens
1413             middle.replace(middle.children[1])
1414
1415         return False
1416
1417     return True
1418
1419
1420 def should_split_line(line: Line, opening_bracket: Leaf) -> bool:
1421     """Should `line` be immediately split with `delimiter_split()` after RHS?"""
1422
1423     if not (opening_bracket.parent and opening_bracket.value in "[{("):
1424         return False
1425
1426     # We're essentially checking if the body is delimited by commas and there's more
1427     # than one of them (we're excluding the trailing comma and if the delimiter priority
1428     # is still commas, that means there's more).
1429     exclude = set()
1430     trailing_comma = False
1431     try:
1432         last_leaf = line.leaves[-1]
1433         if last_leaf.type == token.COMMA:
1434             trailing_comma = True
1435             exclude.add(id(last_leaf))
1436         max_priority = line.bracket_tracker.max_delimiter_priority(exclude=exclude)
1437     except (IndexError, ValueError):
1438         return False
1439
1440     return max_priority == COMMA_PRIORITY and (
1441         (line.mode.magic_trailing_comma and trailing_comma)
1442         # always explode imports
1443         or opening_bracket.parent.type in {syms.atom, syms.import_from}
1444     )
1445
1446
1447 def generate_trailers_to_omit(line: Line, line_length: int) -> Iterator[Set[LeafID]]:
1448     """Generate sets of closing bracket IDs that should be omitted in a RHS.
1449
1450     Brackets can be omitted if the entire trailer up to and including
1451     a preceding closing bracket fits in one line.
1452
1453     Yielded sets are cumulative (contain results of previous yields, too).  First
1454     set is empty, unless the line should explode, in which case bracket pairs until
1455     the one that needs to explode are omitted.
1456     """
1457
1458     omit: Set[LeafID] = set()
1459     if not line.magic_trailing_comma:
1460         yield omit
1461
1462     length = 4 * line.depth
1463     opening_bracket: Optional[Leaf] = None
1464     closing_bracket: Optional[Leaf] = None
1465     inner_brackets: Set[LeafID] = set()
1466     for index, leaf, leaf_length in line.enumerate_with_length(reversed=True):
1467         length += leaf_length
1468         if length > line_length:
1469             break
1470
1471         has_inline_comment = leaf_length > len(leaf.value) + len(leaf.prefix)
1472         if leaf.type == STANDALONE_COMMENT or has_inline_comment:
1473             break
1474
1475         if opening_bracket:
1476             if leaf is opening_bracket:
1477                 opening_bracket = None
1478             elif leaf.type in CLOSING_BRACKETS:
1479                 prev = line.leaves[index - 1] if index > 0 else None
1480                 if (
1481                     prev
1482                     and prev.type == token.COMMA
1483                     and leaf.opening_bracket is not None
1484                     and not is_one_sequence_between(
1485                         leaf.opening_bracket, leaf, line.leaves
1486                     )
1487                 ):
1488                     # Never omit bracket pairs with trailing commas.
1489                     # We need to explode on those.
1490                     break
1491
1492                 inner_brackets.add(id(leaf))
1493         elif leaf.type in CLOSING_BRACKETS:
1494             prev = line.leaves[index - 1] if index > 0 else None
1495             if prev and prev.type in OPENING_BRACKETS:
1496                 # Empty brackets would fail a split so treat them as "inner"
1497                 # brackets (e.g. only add them to the `omit` set if another
1498                 # pair of brackets was good enough.
1499                 inner_brackets.add(id(leaf))
1500                 continue
1501
1502             if closing_bracket:
1503                 omit.add(id(closing_bracket))
1504                 omit.update(inner_brackets)
1505                 inner_brackets.clear()
1506                 yield omit
1507
1508             if (
1509                 prev
1510                 and prev.type == token.COMMA
1511                 and leaf.opening_bracket is not None
1512                 and not is_one_sequence_between(leaf.opening_bracket, leaf, line.leaves)
1513             ):
1514                 # Never omit bracket pairs with trailing commas.
1515                 # We need to explode on those.
1516                 break
1517
1518             if leaf.value:
1519                 opening_bracket = leaf.opening_bracket
1520                 closing_bracket = leaf
1521
1522
1523 def run_transformer(
1524     line: Line,
1525     transform: Transformer,
1526     mode: Mode,
1527     features: Collection[Feature],
1528     *,
1529     line_str: str = "",
1530 ) -> List[Line]:
1531     if not line_str:
1532         line_str = line_to_string(line)
1533     result: List[Line] = []
1534     for transformed_line in transform(line, features, mode):
1535         if str(transformed_line).strip("\n") == line_str:
1536             raise CannotTransform("Line transformer returned an unchanged result")
1537
1538         result.extend(transform_line(transformed_line, mode=mode, features=features))
1539
1540     features_set = set(features)
1541     if (
1542         Feature.FORCE_OPTIONAL_PARENTHESES in features_set
1543         or transform.__class__.__name__ != "rhs"
1544         or not line.bracket_tracker.invisible
1545         or any(bracket.value for bracket in line.bracket_tracker.invisible)
1546         or line.contains_multiline_strings()
1547         or result[0].contains_uncollapsable_type_comments()
1548         or result[0].contains_unsplittable_type_ignore()
1549         or is_line_short_enough(result[0], mode=mode)
1550         # If any leaves have no parents (which _can_ occur since
1551         # `transform(line)` potentially destroys the line's underlying node
1552         # structure), then we can't proceed. Doing so would cause the below
1553         # call to `append_leaves()` to fail.
1554         or any(leaf.parent is None for leaf in line.leaves)
1555     ):
1556         return result
1557
1558     line_copy = line.clone()
1559     append_leaves(line_copy, line, line.leaves)
1560     features_fop = features_set | {Feature.FORCE_OPTIONAL_PARENTHESES}
1561     second_opinion = run_transformer(
1562         line_copy, transform, mode, features_fop, line_str=line_str
1563     )
1564     if all(is_line_short_enough(ln, mode=mode) for ln in second_opinion):
1565         result = second_opinion
1566     return result