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.
   8 from contextlib import contextmanager
 
   9 from dataclasses import replace
 
  10 from datetime import datetime
 
  12 from json.decoder import JSONDecodeError
 
  13 from pathlib import Path
 
  31 from click.core import ParameterSource
 
  32 from mypy_extensions import mypyc_attr
 
  33 from pathspec import PathSpec
 
  34 from pathspec.patterns.gitwildmatch import GitWildMatchPatternError
 
  36 from _black_version import version as __version__
 
  37 from black.cache import Cache, get_cache_info, read_cache, write_cache
 
  38 from black.comments import normalize_fmt_off
 
  39 from black.const import (
 
  45 from black.files import (
 
  48     find_user_pyproject_toml,
 
  51     normalize_path_maybe_ignore,
 
  53     wrap_stream_for_windows,
 
  55 from black.handle_ipynb_magics import (
 
  58     jupyter_dependencies_are_installed,
 
  60     put_trailing_semicolon_back,
 
  61     remove_trailing_semicolon,
 
  64 from black.linegen import LN, LineGenerator, transform_line
 
  65 from black.lines import EmptyLineTracker, LinesBlock
 
  66 from black.mode import (
 
  67     FUTURE_FLAG_TO_FEATURE,
 
  74 from black.nodes import (
 
  77     is_simple_decorator_expression,
 
  81 from black.output import color_diff, diff, dump_to_file, err, ipynb_diff, out
 
  82 from black.parsing import InvalidInput  # noqa F401
 
  83 from black.parsing import lib2to3_parse, parse_ast, stringify_ast
 
  84 from black.report import Changed, NothingChanged, Report
 
  85 from black.trans import iter_fexpr_spans
 
  86 from blib2to3.pgen2 import token
 
  87 from blib2to3.pytree import Leaf, Node
 
  89 COMPILED = Path(__file__).suffix in (".pyd", ".so")
 
  97 class WriteBack(Enum):
 
 105     def from_configuration(
 
 106         cls, *, check: bool, diff: bool, color: bool = False
 
 108         if check and not diff:
 
 112             return cls.COLOR_DIFF
 
 114         return cls.DIFF if diff else cls.YES
 
 117 # Legacy name, left for integrations.
 
 121 def read_pyproject_toml(
 
 122     ctx: click.Context, param: click.Parameter, value: Optional[str]
 
 124     """Inject Black configuration from "pyproject.toml" into defaults in `ctx`.
 
 126     Returns the path to a successfully found and read configuration file, None
 
 130         value = find_pyproject_toml(ctx.params.get("src", ()))
 
 135         config = parse_pyproject_toml(value)
 
 136     except (OSError, ValueError) as e:
 
 137         raise click.FileError(
 
 138             filename=value, hint=f"Error reading configuration file: {e}"
 
 144         # Sanitize the values to be Click friendly. For more information please see:
 
 145         # https://github.com/psf/black/issues/1458
 
 146         # https://github.com/pallets/click/issues/1567
 
 148             k: str(v) if not isinstance(v, (list, dict)) else v
 
 149             for k, v in config.items()
 
 152     target_version = config.get("target_version")
 
 153     if target_version is not None and not isinstance(target_version, list):
 
 154         raise click.BadOptionUsage(
 
 155             "target-version", "Config key target-version must be a list"
 
 158     default_map: Dict[str, Any] = {}
 
 160         default_map.update(ctx.default_map)
 
 161     default_map.update(config)
 
 163     ctx.default_map = default_map
 
 167 def target_version_option_callback(
 
 168     c: click.Context, p: Union[click.Option, click.Parameter], v: Tuple[str, ...]
 
 169 ) -> List[TargetVersion]:
 
 170     """Compute the target versions from a --target-version flag.
 
 172     This is its own function because mypy couldn't infer the type correctly
 
 173     when it was a lambda, causing mypyc trouble.
 
 175     return [TargetVersion[val.upper()] for val in v]
 
 178 def re_compile_maybe_verbose(regex: str) -> Pattern[str]:
 
 179     """Compile a regular expression string in `regex`.
 
 181     If it contains newlines, use verbose mode.
 
 184         regex = "(?x)" + regex
 
 185     compiled: Pattern[str] = re.compile(regex)
 
 191     param: click.Parameter,
 
 192     value: Optional[str],
 
 193 ) -> Optional[Pattern[str]]:
 
 195         return re_compile_maybe_verbose(value) if value is not None else None
 
 196     except re.error as e:
 
 197         raise click.BadParameter(f"Not a valid regular expression: {e}") from None
 
 201     context_settings={"help_option_names": ["-h", "--help"]},
 
 202     # While Click does set this field automatically using the docstring, mypyc
 
 203     # (annoyingly) strips 'em so we need to set it here too.
 
 204     help="The uncompromising code formatter.",
 
 206 @click.option("-c", "--code", type=str, help="Format the code passed in as a string.")
 
 211     default=DEFAULT_LINE_LENGTH,
 
 212     help="How many characters per line to allow.",
 
 218     type=click.Choice([v.name.lower() for v in TargetVersion]),
 
 219     callback=target_version_option_callback,
 
 222         "Python versions that should be supported by Black's output. [default: per-file"
 
 230         "Format all input files like typing stubs regardless of file extension (useful"
 
 231         " when piping source on standard input)."
 
 238         "Format all input files like Jupyter Notebooks regardless of file extension "
 
 239         "(useful when piping source on standard input)."
 
 243     "--python-cell-magics",
 
 246         "When processing Jupyter Notebooks, add the given magic to the list"
 
 247         f" of known python-magics ({', '.join(PYTHON_CELL_MAGICS)})."
 
 248         " Useful for formatting cells with custom python magics."
 
 254     "--skip-source-first-line",
 
 256     help="Skip the first line of the source code.",
 
 260     "--skip-string-normalization",
 
 262     help="Don't normalize string quotes or prefixes.",
 
 266     "--skip-magic-trailing-comma",
 
 268     help="Don't use trailing commas as a reason to split lines.",
 
 271     "--experimental-string-processing",
 
 274     help="(DEPRECATED and now included in --preview) Normalize string literals.",
 
 280         "Enable potentially disruptive style changes that may be added to Black's main"
 
 281         " functionality in the next major release."
 
 288         "Don't write the files back, just return the status. Return code 0 means"
 
 289         " nothing would change. Return code 1 means some files would be reformatted."
 
 290         " Return code 123 means there was an internal error."
 
 296     help="Don't write the files back, just output a diff for each file on stdout.",
 
 299     "--color/--no-color",
 
 301     help="Show colored diff. Only applies when `--diff` is given.",
 
 306     help="If --fast given, skip temporary sanity checks. [default: --safe]",
 
 309     "--required-version",
 
 312         "Require a specific version of Black to be running (useful for unifying results"
 
 313         " across many environments e.g. with a pyproject.toml file). It can be"
 
 314         " either a major version number or an exact version."
 
 320     default=DEFAULT_INCLUDES,
 
 321     callback=validate_regex,
 
 323         "A regular expression that matches files and directories that should be"
 
 324         " included on recursive searches. An empty value means all files are included"
 
 325         " regardless of the name. Use forward slashes for directories on all platforms"
 
 326         " (Windows, too). Exclusions are calculated first, inclusions later."
 
 333     callback=validate_regex,
 
 335         "A regular expression that matches files and directories that should be"
 
 336         " excluded on recursive searches. An empty value means no paths are excluded."
 
 337         " Use forward slashes for directories on all platforms (Windows, too)."
 
 338         " Exclusions are calculated first, inclusions later. [default:"
 
 339         f" {DEFAULT_EXCLUDES}]"
 
 346     callback=validate_regex,
 
 348         "Like --exclude, but adds additional files and directories on top of the"
 
 349         " excluded ones. (Useful if you simply want to add to the default)"
 
 355     callback=validate_regex,
 
 357         "Like --exclude, but files and directories matching this regex will be "
 
 358         "excluded even when they are passed explicitly as arguments."
 
 365         "The name of the file when passing it through stdin. Useful to make "
 
 366         "sure Black will respect --force-exclude option on some "
 
 367         "editors that rely on using stdin."
 
 373     type=click.IntRange(min=1),
 
 375     help="Number of parallel workers [default: number of CPUs in the system]",
 
 382         "Don't emit non-error messages to stderr. Errors are still emitted; silence"
 
 383         " those with 2>/dev/null."
 
 391         "Also emit messages to stderr about files that were not changed or were ignored"
 
 392         " due to exclusion patterns."
 
 395 @click.version_option(
 
 398         f"%(prog)s, %(version)s (compiled: {'yes' if COMPILED else 'no'})\n"
 
 399         f"Python ({platform.python_implementation()}) {platform.python_version()}"
 
 406         exists=True, file_okay=True, dir_okay=True, readable=True, allow_dash=True
 
 422     callback=read_pyproject_toml,
 
 423     help="Read configuration from FILE path.",
 
 426 def main(  # noqa: C901
 
 430     target_version: List[TargetVersion],
 
 437     python_cell_magics: Sequence[str],
 
 438     skip_source_first_line: bool,
 
 439     skip_string_normalization: bool,
 
 440     skip_magic_trailing_comma: bool,
 
 441     experimental_string_processing: bool,
 
 445     required_version: Optional[str],
 
 446     include: Pattern[str],
 
 447     exclude: Optional[Pattern[str]],
 
 448     extend_exclude: Optional[Pattern[str]],
 
 449     force_exclude: Optional[Pattern[str]],
 
 450     stdin_filename: Optional[str],
 
 451     workers: Optional[int],
 
 452     src: Tuple[str, ...],
 
 453     config: Optional[str],
 
 455     """The uncompromising code formatter."""
 
 456     ctx.ensure_object(dict)
 
 458     if src and code is not None:
 
 461             + "\n\n'SRC' and 'code' cannot be passed simultaneously."
 
 464     if not src and code is None:
 
 465         out(main.get_usage(ctx) + "\n\nOne of 'SRC' or 'code' is required.")
 
 469         find_project_root(src, stdin_filename) if code is None else (None, None)
 
 471     ctx.obj["root"] = root
 
 476                 f"Identified `{root}` as project root containing a {method}.",
 
 484                     else (normalize_path_maybe_ignore(Path(source), root), source)
 
 488             srcs_string = ", ".join(
 
 493                         else f'\033[31m"{source} (skipping - invalid)"\033[34m'
 
 495                     for _norm, source in normalized
 
 498             out(f"Sources to be formatted: {srcs_string}", fg="blue")
 
 501             config_source = ctx.get_parameter_source("config")
 
 502             user_level_config = str(find_user_pyproject_toml())
 
 503             if config == user_level_config:
 
 506                         "Using configuration from user-level config at "
 
 507                         f"'{user_level_config}'."
 
 511             elif config_source in (
 
 512                 ParameterSource.DEFAULT,
 
 513                 ParameterSource.DEFAULT_MAP,
 
 515                 out("Using configuration from project root.", fg="blue")
 
 517                 out(f"Using configuration in '{config}'.", fg="blue")
 
 519                 for param, value in ctx.default_map.items():
 
 520                     out(f"{param}: {value}")
 
 522     error_msg = "Oh no! 💥 💔 💥"
 
 525         and required_version != __version__
 
 526         and required_version != __version__.split(".")[0]
 
 529             f"{error_msg} The required version `{required_version}` does not match"
 
 530             f" the running version `{__version__}`!"
 
 534         err("Cannot pass both `pyi` and `ipynb` flags!")
 
 537     write_back = WriteBack.from_configuration(check=check, diff=diff, color=color)
 
 539         versions = set(target_version)
 
 541         # We'll autodetect later.
 
 544         target_versions=versions,
 
 545         line_length=line_length,
 
 548         skip_source_first_line=skip_source_first_line,
 
 549         string_normalization=not skip_string_normalization,
 
 550         magic_trailing_comma=not skip_magic_trailing_comma,
 
 551         experimental_string_processing=experimental_string_processing,
 
 553         python_cell_magics=set(python_cell_magics),
 
 557         # Run in quiet mode by default with -c; the extra output isn't useful.
 
 558         # You can still pass -v to get verbose output.
 
 561     report = Report(check=check, diff=diff, quiet=quiet, verbose=verbose)
 
 565             content=code, fast=fast, write_back=write_back, mode=mode, report=report
 
 569             sources = get_sources(
 
 576                 extend_exclude=extend_exclude,
 
 577                 force_exclude=force_exclude,
 
 579                 stdin_filename=stdin_filename,
 
 581         except GitWildMatchPatternError:
 
 586             "No Python files are present to be formatted. Nothing to do 😴",
 
 592         if len(sources) == 1:
 
 596                 write_back=write_back,
 
 601             from black.concurrency import reformat_many
 
 606                 write_back=write_back,
 
 612     if verbose or not quiet:
 
 613         if code is None and (verbose or report.change_count or report.failure_count):
 
 615         out(error_msg if report.return_code else "All done! ✨ 🍰 ✨")
 
 617             click.echo(str(report), err=True)
 
 618     ctx.exit(report.return_code)
 
 624     src: Tuple[str, ...],
 
 627     include: Pattern[str],
 
 628     exclude: Optional[Pattern[str]],
 
 629     extend_exclude: Optional[Pattern[str]],
 
 630     force_exclude: Optional[Pattern[str]],
 
 632     stdin_filename: Optional[str],
 
 634     """Compute the set of files to be formatted."""
 
 635     sources: Set[Path] = set()
 
 636     root = ctx.obj["root"]
 
 638     using_default_exclude = exclude is None
 
 639     exclude = re_compile_maybe_verbose(DEFAULT_EXCLUDES) if exclude is None else exclude
 
 640     gitignore: Optional[Dict[Path, PathSpec]] = None
 
 641     root_gitignore = get_gitignore(root)
 
 644         if s == "-" and stdin_filename:
 
 645             p = Path(stdin_filename)
 
 651         if is_stdin or p.is_file():
 
 652             normalized_path = normalize_path_maybe_ignore(p, ctx.obj["root"], report)
 
 653             if normalized_path is None:
 
 656             normalized_path = "/" + normalized_path
 
 657             # Hard-exclude any files that matches the `--force-exclude` regex.
 
 659                 force_exclude_match = force_exclude.search(normalized_path)
 
 661                 force_exclude_match = None
 
 662             if force_exclude_match and force_exclude_match.group(0):
 
 663                 report.path_ignored(p, "matches the --force-exclude regular expression")
 
 667                 p = Path(f"{STDIN_PLACEHOLDER}{str(p)}")
 
 669             if p.suffix == ".ipynb" and not jupyter_dependencies_are_installed(
 
 670                 verbose=verbose, quiet=quiet
 
 676             p = root / normalize_path_maybe_ignore(p, ctx.obj["root"], report)
 
 677             if using_default_exclude:
 
 679                     root: root_gitignore,
 
 699             err(f"invalid path: {s}")
 
 704     src: Sized, msg: str, quiet: bool, verbose: bool, ctx: click.Context
 
 707     Exit if there is no `src` provided for formatting
 
 710         if verbose or not quiet:
 
 716     content: str, fast: bool, write_back: WriteBack, mode: Mode, report: Report
 
 719     Reformat and print out `content` without spawning child processes.
 
 720     Similar to `reformat_one`, but for string content.
 
 722     `fast`, `write_back`, and `mode` options are passed to
 
 723     :func:`format_file_in_place` or :func:`format_stdin_to_stdout`.
 
 725     path = Path("<string>")
 
 728         if format_stdin_to_stdout(
 
 729             content=content, fast=fast, write_back=write_back, mode=mode
 
 731             changed = Changed.YES
 
 732         report.done(path, changed)
 
 733     except Exception as exc:
 
 735             traceback.print_exc()
 
 736         report.failed(path, str(exc))
 
 739 # diff-shades depends on being to monkeypatch this function to operate. I know it's
 
 740 # not ideal, but this shouldn't cause any issues ... hopefully. ~ichard26
 
 741 @mypyc_attr(patchable=True)
 
 743     src: Path, fast: bool, write_back: WriteBack, mode: Mode, report: "Report"
 
 745     """Reformat a single file under `src` without spawning child processes.
 
 747     `fast`, `write_back`, and `mode` options are passed to
 
 748     :func:`format_file_in_place` or :func:`format_stdin_to_stdout`.
 
 755         elif str(src).startswith(STDIN_PLACEHOLDER):
 
 757             # Use the original name again in case we want to print something
 
 759             src = Path(str(src)[len(STDIN_PLACEHOLDER) :])
 
 764             if src.suffix == ".pyi":
 
 765                 mode = replace(mode, is_pyi=True)
 
 766             elif src.suffix == ".ipynb":
 
 767                 mode = replace(mode, is_ipynb=True)
 
 768             if format_stdin_to_stdout(fast=fast, write_back=write_back, mode=mode):
 
 769                 changed = Changed.YES
 
 772             if write_back not in (WriteBack.DIFF, WriteBack.COLOR_DIFF):
 
 773                 cache = read_cache(mode)
 
 774                 res_src = src.resolve()
 
 775                 res_src_s = str(res_src)
 
 776                 if res_src_s in cache and cache[res_src_s] == get_cache_info(res_src):
 
 777                     changed = Changed.CACHED
 
 778             if changed is not Changed.CACHED and format_file_in_place(
 
 779                 src, fast=fast, write_back=write_back, mode=mode
 
 781                 changed = Changed.YES
 
 782             if (write_back is WriteBack.YES and changed is not Changed.CACHED) or (
 
 783                 write_back is WriteBack.CHECK and changed is Changed.NO
 
 785                 write_cache(cache, [src], mode)
 
 786         report.done(src, changed)
 
 787     except Exception as exc:
 
 789             traceback.print_exc()
 
 790         report.failed(src, str(exc))
 
 793 def format_file_in_place(
 
 797     write_back: WriteBack = WriteBack.NO,
 
 798     lock: Any = None,  # multiprocessing.Manager().Lock() is some crazy proxy
 
 800     """Format file under `src` path. Return True if changed.
 
 802     If `write_back` is DIFF, write a diff to stdout. If it is YES, write reformatted
 
 804     `mode` and `fast` options are passed to :func:`format_file_contents`.
 
 806     if src.suffix == ".pyi":
 
 807         mode = replace(mode, is_pyi=True)
 
 808     elif src.suffix == ".ipynb":
 
 809         mode = replace(mode, is_ipynb=True)
 
 811     then = datetime.utcfromtimestamp(src.stat().st_mtime)
 
 813     with open(src, "rb") as buf:
 
 814         if mode.skip_source_first_line:
 
 815             header = buf.readline()
 
 816         src_contents, encoding, newline = decode_bytes(buf.read())
 
 818         dst_contents = format_file_contents(src_contents, fast=fast, mode=mode)
 
 819     except NothingChanged:
 
 821     except JSONDecodeError:
 
 823             f"File '{src}' cannot be parsed as valid Jupyter notebook."
 
 825     src_contents = header.decode(encoding) + src_contents
 
 826     dst_contents = header.decode(encoding) + dst_contents
 
 828     if write_back == WriteBack.YES:
 
 829         with open(src, "w", encoding=encoding, newline=newline) as f:
 
 830             f.write(dst_contents)
 
 831     elif write_back in (WriteBack.DIFF, WriteBack.COLOR_DIFF):
 
 832         now = datetime.utcnow()
 
 833         src_name = f"{src}\t{then} +0000"
 
 834         dst_name = f"{src}\t{now} +0000"
 
 836             diff_contents = ipynb_diff(src_contents, dst_contents, src_name, dst_name)
 
 838             diff_contents = diff(src_contents, dst_contents, src_name, dst_name)
 
 840         if write_back == WriteBack.COLOR_DIFF:
 
 841             diff_contents = color_diff(diff_contents)
 
 843         with lock or nullcontext():
 
 844             f = io.TextIOWrapper(
 
 850             f = wrap_stream_for_windows(f)
 
 851             f.write(diff_contents)
 
 857 def format_stdin_to_stdout(
 
 860     content: Optional[str] = None,
 
 861     write_back: WriteBack = WriteBack.NO,
 
 864     """Format file on stdin. Return True if changed.
 
 866     If content is None, it's read from sys.stdin.
 
 868     If `write_back` is YES, write reformatted code back to stdout. If it is DIFF,
 
 869     write a diff to stdout. The `mode` argument is passed to
 
 870     :func:`format_file_contents`.
 
 872     then = datetime.utcnow()
 
 875         src, encoding, newline = decode_bytes(sys.stdin.buffer.read())
 
 877         src, encoding, newline = content, "utf-8", ""
 
 881         dst = format_file_contents(src, fast=fast, mode=mode)
 
 884     except NothingChanged:
 
 888         f = io.TextIOWrapper(
 
 889             sys.stdout.buffer, encoding=encoding, newline=newline, write_through=True
 
 891         if write_back == WriteBack.YES:
 
 892             # Make sure there's a newline after the content
 
 893             if dst and dst[-1] != "\n":
 
 896         elif write_back in (WriteBack.DIFF, WriteBack.COLOR_DIFF):
 
 897             now = datetime.utcnow()
 
 898             src_name = f"STDIN\t{then} +0000"
 
 899             dst_name = f"STDOUT\t{now} +0000"
 
 900             d = diff(src, dst, src_name, dst_name)
 
 901             if write_back == WriteBack.COLOR_DIFF:
 
 903                 f = wrap_stream_for_windows(f)
 
 908 def check_stability_and_equivalence(
 
 909     src_contents: str, dst_contents: str, *, mode: Mode
 
 911     """Perform stability and equivalence checks.
 
 913     Raise AssertionError if source and destination contents are not
 
 914     equivalent, or if a second pass of the formatter would format the
 
 917     assert_equivalent(src_contents, dst_contents)
 
 918     assert_stable(src_contents, dst_contents, mode=mode)
 
 921 def format_file_contents(src_contents: str, *, fast: bool, mode: Mode) -> FileContent:
 
 922     """Reformat contents of a file and return new contents.
 
 924     If `fast` is False, additionally confirm that the reformatted code is
 
 925     valid by calling :func:`assert_equivalent` and :func:`assert_stable` on it.
 
 926     `mode` is passed to :func:`format_str`.
 
 928     if not mode.preview and not src_contents.strip():
 
 932         dst_contents = format_ipynb_string(src_contents, fast=fast, mode=mode)
 
 934         dst_contents = format_str(src_contents, mode=mode)
 
 935     if src_contents == dst_contents:
 
 938     if not fast and not mode.is_ipynb:
 
 939         # Jupyter notebooks will already have been checked above.
 
 940         check_stability_and_equivalence(src_contents, dst_contents, mode=mode)
 
 944 def validate_cell(src: str, mode: Mode) -> None:
 
 945     """Check that cell does not already contain TransformerManager transformations,
 
 946     or non-Python cell magics, which might cause tokenizer_rt to break because of
 
 949     If a cell contains ``!ls``, then it'll be transformed to
 
 950     ``get_ipython().system('ls')``. However, if the cell originally contained
 
 951     ``get_ipython().system('ls')``, then it would get transformed in the same way:
 
 953         >>> TransformerManager().transform_cell("get_ipython().system('ls')")
 
 954         "get_ipython().system('ls')\n"
 
 955         >>> TransformerManager().transform_cell("!ls")
 
 956         "get_ipython().system('ls')\n"
 
 958     Due to the impossibility of safely roundtripping in such situations, cells
 
 959     containing transformed magics will be ignored.
 
 961     if any(transformed_magic in src for transformed_magic in TRANSFORMED_MAGICS):
 
 965         and src.split()[0][2:] not in PYTHON_CELL_MAGICS | mode.python_cell_magics
 
 970 def format_cell(src: str, *, fast: bool, mode: Mode) -> str:
 
 971     """Format code in given cell of Jupyter notebook.
 
 975       - if cell has trailing semicolon, remove it;
 
 976       - if cell has IPython magics, mask them;
 
 978       - reinstate IPython magics;
 
 979       - reinstate trailing semicolon (if originally present);
 
 980       - strip trailing newlines.
 
 982     Cells with syntax errors will not be processed, as they
 
 983     could potentially be automagics or multi-line magics, which
 
 984     are currently not supported.
 
 986     validate_cell(src, mode)
 
 987     src_without_trailing_semicolon, has_trailing_semicolon = remove_trailing_semicolon(
 
 991         masked_src, replacements = mask_cell(src_without_trailing_semicolon)
 
 993         raise NothingChanged from None
 
 994     masked_dst = format_str(masked_src, mode=mode)
 
 996         check_stability_and_equivalence(masked_src, masked_dst, mode=mode)
 
 997     dst_without_trailing_semicolon = unmask_cell(masked_dst, replacements)
 
 998     dst = put_trailing_semicolon_back(
 
 999         dst_without_trailing_semicolon, has_trailing_semicolon
 
1001     dst = dst.rstrip("\n")
 
1003         raise NothingChanged from None
 
1007 def validate_metadata(nb: MutableMapping[str, Any]) -> None:
 
1008     """If notebook is marked as non-Python, don't format it.
 
1010     All notebook metadata fields are optional, see
 
1011     https://nbformat.readthedocs.io/en/latest/format_description.html. So
 
1012     if a notebook has empty metadata, we will try to parse it anyway.
 
1014     language = nb.get("metadata", {}).get("language_info", {}).get("name", None)
 
1015     if language is not None and language != "python":
 
1016         raise NothingChanged from None
 
1019 def format_ipynb_string(src_contents: str, *, fast: bool, mode: Mode) -> FileContent:
 
1020     """Format Jupyter notebook.
 
1022     Operate cell-by-cell, only on code cells, only for Python notebooks.
 
1023     If the ``.ipynb`` originally had a trailing newline, it'll be preserved.
 
1025     if mode.preview and not src_contents:
 
1026         raise NothingChanged
 
1028     trailing_newline = src_contents[-1] == "\n"
 
1030     nb = json.loads(src_contents)
 
1031     validate_metadata(nb)
 
1032     for cell in nb["cells"]:
 
1033         if cell.get("cell_type", None) == "code":
 
1035                 src = "".join(cell["source"])
 
1036                 dst = format_cell(src, fast=fast, mode=mode)
 
1037             except NothingChanged:
 
1040                 cell["source"] = dst.splitlines(keepends=True)
 
1043         dst_contents = json.dumps(nb, indent=1, ensure_ascii=False)
 
1044         if trailing_newline:
 
1045             dst_contents = dst_contents + "\n"
 
1048         raise NothingChanged
 
1051 def format_str(src_contents: str, *, mode: Mode) -> str:
 
1052     """Reformat a string and return new contents.
 
1054     `mode` determines formatting options, such as how many characters per line are
 
1058     >>> print(black.format_str("def f(arg:str='')->None:...", mode=black.Mode()))
 
1059     def f(arg: str = "") -> None:
 
1062     A more complex example:
 
1065     ...   black.format_str(
 
1066     ...     "def f(arg:str='')->None: hey",
 
1067     ...     mode=black.Mode(
 
1068     ...       target_versions={black.TargetVersion.PY36},
 
1070     ...       string_normalization=False,
 
1081     dst_contents = _format_str_once(src_contents, mode=mode)
 
1082     # Forced second pass to work around optional trailing commas (becoming
 
1083     # forced trailing commas on pass 2) interacting differently with optional
 
1084     # parentheses.  Admittedly ugly.
 
1085     if src_contents != dst_contents:
 
1086         return _format_str_once(dst_contents, mode=mode)
 
1090 def _format_str_once(src_contents: str, *, mode: Mode) -> str:
 
1091     src_node = lib2to3_parse(src_contents.lstrip(), mode.target_versions)
 
1092     dst_blocks: List[LinesBlock] = []
 
1093     if mode.target_versions:
 
1094         versions = mode.target_versions
 
1096         future_imports = get_future_imports(src_node)
 
1097         versions = detect_target_versions(src_node, future_imports=future_imports)
 
1099     normalize_fmt_off(src_node, preview=mode.preview)
 
1100     lines = LineGenerator(mode=mode)
 
1101     elt = EmptyLineTracker(mode=mode)
 
1102     split_line_features = {
 
1104         for feature in {Feature.TRAILING_COMMA_IN_CALL, Feature.TRAILING_COMMA_IN_DEF}
 
1105         if supports_feature(versions, feature)
 
1107     block: Optional[LinesBlock] = None
 
1108     for current_line in lines.visit(src_node):
 
1109         block = elt.maybe_empty_lines(current_line)
 
1110         dst_blocks.append(block)
 
1111         for line in transform_line(
 
1112             current_line, mode=mode, features=split_line_features
 
1114             block.content_lines.append(str(line))
 
1116         dst_blocks[-1].after = 0
 
1118     for block in dst_blocks:
 
1119         dst_contents.extend(block.all_lines())
 
1120     if mode.preview and not dst_contents:
 
1121         # Use decode_bytes to retrieve the correct source newline (CRLF or LF),
 
1122         # and check if normalized_content has more than one line
 
1123         normalized_content, _, newline = decode_bytes(src_contents.encode("utf-8"))
 
1124         if "\n" in normalized_content:
 
1127     return "".join(dst_contents)
 
1130 def decode_bytes(src: bytes) -> Tuple[FileContent, Encoding, NewLine]:
 
1131     """Return a tuple of (decoded_contents, encoding, newline).
 
1133     `newline` is either CRLF or LF but `decoded_contents` is decoded with
 
1134     universal newlines (i.e. only contains LF).
 
1136     srcbuf = io.BytesIO(src)
 
1137     encoding, lines = tokenize.detect_encoding(srcbuf.readline)
 
1139         return "", encoding, "\n"
 
1141     newline = "\r\n" if b"\r\n" == lines[0][-2:] else "\n"
 
1143     with io.TextIOWrapper(srcbuf, encoding) as tiow:
 
1144         return tiow.read(), encoding, newline
 
1147 def get_features_used(  # noqa: C901
 
1148     node: Node, *, future_imports: Optional[Set[str]] = None
 
1150     """Return a set of (relatively) new Python features used in this file.
 
1152     Currently looking for:
 
1154     - self-documenting expressions in f-strings (f"{x=}");
 
1155     - underscores in numeric literals;
 
1156     - trailing commas after * or ** in function signatures and calls;
 
1157     - positional only arguments in function signatures and lambdas;
 
1158     - assignment expression;
 
1159     - relaxed decorator syntax;
 
1160     - usage of __future__ flags (annotations);
 
1161     - print / exec statements;
 
1163     features: Set[Feature] = set()
 
1166             FUTURE_FLAG_TO_FEATURE[future_import]
 
1167             for future_import in future_imports
 
1168             if future_import in FUTURE_FLAG_TO_FEATURE
 
1171     for n in node.pre_order():
 
1172         if is_string_token(n):
 
1173             value_head = n.value[:2]
 
1174             if value_head in {'f"', 'F"', "f'", "F'", "rf", "fr", "RF", "FR"}:
 
1175                 features.add(Feature.F_STRINGS)
 
1176                 if Feature.DEBUG_F_STRINGS not in features:
 
1177                     for span_beg, span_end in iter_fexpr_spans(n.value):
 
1178                         if n.value[span_beg : span_end - 1].rstrip().endswith("="):
 
1179                             features.add(Feature.DEBUG_F_STRINGS)
 
1182         elif is_number_token(n):
 
1184                 features.add(Feature.NUMERIC_UNDERSCORES)
 
1186         elif n.type == token.SLASH:
 
1187             if n.parent and n.parent.type in {
 
1192                 features.add(Feature.POS_ONLY_ARGUMENTS)
 
1194         elif n.type == token.COLONEQUAL:
 
1195             features.add(Feature.ASSIGNMENT_EXPRESSIONS)
 
1197         elif n.type == syms.decorator:
 
1198             if len(n.children) > 1 and not is_simple_decorator_expression(
 
1201                 features.add(Feature.RELAXED_DECORATORS)
 
1204             n.type in {syms.typedargslist, syms.arglist}
 
1206             and n.children[-1].type == token.COMMA
 
1208             if n.type == syms.typedargslist:
 
1209                 feature = Feature.TRAILING_COMMA_IN_DEF
 
1211                 feature = Feature.TRAILING_COMMA_IN_CALL
 
1213             for ch in n.children:
 
1214                 if ch.type in STARS:
 
1215                     features.add(feature)
 
1217                 if ch.type == syms.argument:
 
1218                     for argch in ch.children:
 
1219                         if argch.type in STARS:
 
1220                             features.add(feature)
 
1223             n.type in {syms.return_stmt, syms.yield_expr}
 
1224             and len(n.children) >= 2
 
1225             and n.children[1].type == syms.testlist_star_expr
 
1226             and any(child.type == syms.star_expr for child in n.children[1].children)
 
1228             features.add(Feature.UNPACKING_ON_FLOW)
 
1231             n.type == syms.annassign
 
1232             and len(n.children) >= 4
 
1233             and n.children[3].type == syms.testlist_star_expr
 
1235             features.add(Feature.ANN_ASSIGN_EXTENDED_RHS)
 
1238             n.type == syms.except_clause
 
1239             and len(n.children) >= 2
 
1240             and n.children[1].type == token.STAR
 
1242             features.add(Feature.EXCEPT_STAR)
 
1244         elif n.type in {syms.subscriptlist, syms.trailer} and any(
 
1245             child.type == syms.star_expr for child in n.children
 
1247             features.add(Feature.VARIADIC_GENERICS)
 
1250             n.type == syms.tname_star
 
1251             and len(n.children) == 3
 
1252             and n.children[2].type == syms.star_expr
 
1254             features.add(Feature.VARIADIC_GENERICS)
 
1259 def detect_target_versions(
 
1260     node: Node, *, future_imports: Optional[Set[str]] = None
 
1261 ) -> Set[TargetVersion]:
 
1262     """Detect the version to target based on the nodes used."""
 
1263     features = get_features_used(node, future_imports=future_imports)
 
1265         version for version in TargetVersion if features <= VERSION_TO_FEATURES[version]
 
1269 def get_future_imports(node: Node) -> Set[str]:
 
1270     """Return a set of __future__ imports in the file."""
 
1271     imports: Set[str] = set()
 
1273     def get_imports_from_children(children: List[LN]) -> Generator[str, None, None]:
 
1274         for child in children:
 
1275             if isinstance(child, Leaf):
 
1276                 if child.type == token.NAME:
 
1279             elif child.type == syms.import_as_name:
 
1280                 orig_name = child.children[0]
 
1281                 assert isinstance(orig_name, Leaf), "Invalid syntax parsing imports"
 
1282                 assert orig_name.type == token.NAME, "Invalid syntax parsing imports"
 
1283                 yield orig_name.value
 
1285             elif child.type == syms.import_as_names:
 
1286                 yield from get_imports_from_children(child.children)
 
1289                 raise AssertionError("Invalid syntax parsing imports")
 
1291     for child in node.children:
 
1292         if child.type != syms.simple_stmt:
 
1295         first_child = child.children[0]
 
1296         if isinstance(first_child, Leaf):
 
1297             # Continue looking if we see a docstring; otherwise stop.
 
1299                 len(child.children) == 2
 
1300                 and first_child.type == token.STRING
 
1301                 and child.children[1].type == token.NEWLINE
 
1307         elif first_child.type == syms.import_from:
 
1308             module_name = first_child.children[1]
 
1309             if not isinstance(module_name, Leaf) or module_name.value != "__future__":
 
1312             imports |= set(get_imports_from_children(first_child.children[3:]))
 
1319 def assert_equivalent(src: str, dst: str) -> None:
 
1320     """Raise AssertionError if `src` and `dst` aren't equivalent."""
 
1322         src_ast = parse_ast(src)
 
1323     except Exception as exc:
 
1324         raise AssertionError(
 
1325             "cannot use --safe with this file; failed to parse source file AST: "
 
1327             "This could be caused by running Black with an older Python version "
 
1328             "that does not support new syntax used in your source file."
 
1332         dst_ast = parse_ast(dst)
 
1333     except Exception as exc:
 
1334         log = dump_to_file("".join(traceback.format_tb(exc.__traceback__)), dst)
 
1335         raise AssertionError(
 
1336             f"INTERNAL ERROR: Black produced invalid code: {exc}. "
 
1337             "Please report a bug on https://github.com/psf/black/issues.  "
 
1338             f"This invalid output might be helpful: {log}"
 
1341     src_ast_str = "\n".join(stringify_ast(src_ast))
 
1342     dst_ast_str = "\n".join(stringify_ast(dst_ast))
 
1343     if src_ast_str != dst_ast_str:
 
1344         log = dump_to_file(diff(src_ast_str, dst_ast_str, "src", "dst"))
 
1345         raise AssertionError(
 
1346             "INTERNAL ERROR: Black produced code that is not equivalent to the"
 
1347             " source.  Please report a bug on "
 
1348             f"https://github.com/psf/black/issues.  This diff might be helpful: {log}"
 
1352 def assert_stable(src: str, dst: str, mode: Mode) -> None:
 
1353     """Raise AssertionError if `dst` reformats differently the second time."""
 
1354     # We shouldn't call format_str() here, because that formats the string
 
1355     # twice and may hide a bug where we bounce back and forth between two
 
1357     newdst = _format_str_once(dst, mode=mode)
 
1361             diff(src, dst, "source", "first pass"),
 
1362             diff(dst, newdst, "first pass", "second pass"),
 
1364         raise AssertionError(
 
1365             "INTERNAL ERROR: Black produced different code on the second pass of the"
 
1366             " formatter.  Please report a bug on https://github.com/psf/black/issues."
 
1367             f"  This diff might be helpful: {log}"
 
1372 def nullcontext() -> Iterator[None]:
 
1373     """Return an empty context manager.
 
1375     To be used like `nullcontext` in Python 3.7.
 
1380 def patch_click() -> None:
 
1381     """Make Click not crash on Python 3.6 with LANG=C.
 
1383     On certain misconfigured environments, Python 3 selects the ASCII encoding as the
 
1384     default which restricts paths that it can access during the lifetime of the
 
1385     application.  Click refuses to work in this scenario by raising a RuntimeError.
 
1387     In case of Black the likelihood that non-ASCII characters are going to be used in
 
1388     file paths is minimal since it's Python source code.  Moreover, this crash was
 
1389     spurious on Python 3.7 thanks to PEP 538 and PEP 540.
 
1391     modules: List[Any] = []
 
1393         from click import core
 
1397         modules.append(core)
 
1399         # Removed in Click 8.1.0 and newer; we keep this around for users who have
 
1400         # older versions installed.
 
1401         from click import _unicodefun  # type: ignore
 
1405         modules.append(_unicodefun)
 
1407     for module in modules:
 
1408         if hasattr(module, "_verify_python3_env"):
 
1409             module._verify_python3_env = lambda: None
 
1410         if hasattr(module, "_verify_python_env"):
 
1411             module._verify_python_env = lambda: None
 
1414 def patched_main() -> None:
 
1415     # PyInstaller patches multiprocessing to need freeze_support() even in non-Windows
 
1416     # environments so just assume we always need to call it if frozen.
 
1417     if getattr(sys, "frozen", False):
 
1418         from multiprocessing import freeze_support
 
1426 if __name__ == "__main__":