Skip to content

Commit

Permalink
fix: improve Black import error [optional-black-improvement]
Browse files Browse the repository at this point in the history
  • Loading branch information
akaihola committed Jan 1, 2025
1 parent dc7cb4d commit e8ca107
Show file tree
Hide file tree
Showing 5 changed files with 60 additions and 27 deletions.
32 changes: 11 additions & 21 deletions src/darker/formatters/black_formatter.py
Original file line number Diff line number Diff line change
Expand Up @@ -93,24 +93,10 @@ def read_config(self, src: tuple[str, ...], args: Namespace) -> None:
def _read_config_file(self, config_path: str) -> None: # noqa: C901
# Local import so Darker can be run without Black installed.
# Do error handling here. This is the first Black importing method being hit.
try:
from black import ( # pylint: disable=import-outside-toplevel
parse_pyproject_toml,
re_compile_maybe_verbose,
)
except ImportError as exc:
logger.warning(
"To re-format code using Black, install it using e.g."
" `pip install 'darker[black]'` or"
" `pip install black`"
)
logger.warning(
"To use a different formatter or no formatter, select it on the"
" command line (e.g. `--formatter=none`) or configuration"
" (e.g. `formatter=none`)"
)
message = "Can't find the Black package"
raise DependencyError(message) from exc
from darker.formatters.black_wrapper import ( # pylint: disable=import-outside-toplevel

Check failure on line 96 in src/darker/formatters/black_formatter.py

View workflow job for this annotation

GitHub Actions / flake8

line too long (96 > 88 characters)
parse_pyproject_toml,
re_compile_maybe_verbose,
)

raw_config = parse_pyproject_toml(config_path)
if "line_length" in raw_config:
Expand Down Expand Up @@ -171,7 +157,9 @@ def run(self, content: TextDocument) -> TextDocument:
"""
# Local import so Darker can be run without Black installed.
# No need for error handling, already done in `BlackFormatter.read_config`.
from black import format_str # pylint: disable=import-outside-toplevel
from darker.formatters.black_wrapper import (

Check failure on line 160 in src/darker/formatters/black_formatter.py

View workflow job for this annotation

GitHub Actions / Pylint

src/darker/formatters/black_formatter.py#L160

Import outside toplevel (darker.formatters.black_wrapper.format_str) (import-outside-toplevel, C0415)
format_str,
) # pylint: disable=import-outside-toplevel

contents_for_black = content.string_with_newline("\n")
if contents_for_black.strip():
Expand All @@ -196,8 +184,10 @@ def _make_black_options(self) -> Mode:

# Local import so Darker can be run without Black installed.
# No need for error handling, already done in `BlackFormatter.read_config`.
from black import FileMode as Mode # pylint: disable=import-outside-toplevel
from black import TargetVersion # pylint: disable=import-outside-toplevel
from darker.formatters.black_wrapper import (

Check failure on line 187 in src/darker/formatters/black_formatter.py

View workflow job for this annotation

GitHub Actions / Pylint

src/darker/formatters/black_formatter.py#L187

Import outside toplevel (darker.formatters.black_wrapper.FileMode, darker.formatters.black_wrapper.TargetVersion) (import-outside-toplevel, C0415)
FileMode as Mode,
TargetVersion,
) # pylint: disable=import-outside-toplevel

mode = BlackModeAttributes()
if "line_length" in self.config:
Expand Down
40 changes: 40 additions & 0 deletions src/darker/formatters/black_wrapper.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,40 @@
"""Attempt to import Black internals needed by the Black formatter plugin."""

import logging

from darker.exceptions import DependencyError

logger = logging.getLogger(__name__)

try:
import black

Check failure on line 10 in src/darker/formatters/black_wrapper.py

View workflow job for this annotation

GitHub Actions / flake8

'black' imported but unused

Check failure on line 10 in src/darker/formatters/black_wrapper.py

View workflow job for this annotation

GitHub Actions / Pylint

src/darker/formatters/black_wrapper.py#L10

Unused import black (unused-import, W0611)
except ImportError as exc:
logger.warning(
"To re-format code using Black, install it using e.g."
" `pip install 'darker[black]'` or"
" `pip install black`"
)
logger.warning(
"To use a different formatter or no formatter, select it on the"
" command line (e.g. `--formatter=none`) or configuration"
" (e.g. `formatter=none`)"
)
message = "Can't find the Black package"

Check failure on line 22 in src/darker/formatters/black_wrapper.py

View workflow job for this annotation

GitHub Actions / Pylint

src/darker/formatters/black_wrapper.py#L22

Constant name "message" doesn't conform to UPPER_CASE naming style (invalid-name, C0103)
raise DependencyError(message) from exc

from black import (

Check failure on line 25 in src/darker/formatters/black_wrapper.py

View workflow job for this annotation

GitHub Actions / flake8

module level import not at top of file

Check failure on line 25 in src/darker/formatters/black_wrapper.py

View workflow job for this annotation

GitHub Actions / Pylint

src/darker/formatters/black_wrapper.py#L25

Import "from black import FileMode, TargetVersion, format_str, parse_pyproject_toml, re_compile_maybe_verbose" should be placed at the top of the module (wrong-import-position, C0413)
FileMode,
TargetVersion,
format_str,
parse_pyproject_toml,
re_compile_maybe_verbose,
)


__all__ = [
"FileMode",
"TargetVersion",
"format_str",

Check failure on line 37 in src/darker/formatters/black_wrapper.py

View workflow job for this annotation

GitHub Actions / isort

src/darker/formatters/black_wrapper.py#L30-L37

re_compile_maybe_verbose, ) - __all__ = [ "FileMode", "TargetVersion",
"parse_pyproject_toml",
"re_compile_maybe_verbose",
]
1 change: 1 addition & 0 deletions src/darker/tests/helpers.py
Original file line number Diff line number Diff line change
Expand Up @@ -28,6 +28,7 @@ def _package_present(
def black_present(*, present: bool) -> Generator[None, None, None]:
"""Context manager to remove or add the ``black`` package temporarily for a test."""
with _package_present("black", present):
del sys.modules["darker.formatters.black_wrapper"]
yield


Expand Down
6 changes: 4 additions & 2 deletions src/darker/tests/test_command_line.py
Original file line number Diff line number Diff line change
Expand Up @@ -589,7 +589,9 @@ def test_black_options(black_options_files, options, expect):
# shared by all test cases. The "main.py" file modified by the test run needs to be
# reset to its original content before the next test case.
black_options_files["main.py"].write_bytes(b'print ("Hello World!")\n')
with patch("black.FileMode", wraps=FileMode) as file_mode_class:
with patch(
"darker.formatters.black_wrapper.FileMode", wraps=FileMode
) as file_mode_class:
# end of test setup, now call the function under test

main(options + [str(path) for path in black_options_files.values()])
Expand Down Expand Up @@ -718,7 +720,7 @@ def test_black_config_file_and_options(
mode_class_mock = Mock(wraps=FileMode)
# Speed up tests by mocking `format_str` to skip running Black
format_str = Mock(return_value="a = [1, 2,]")
with patch("black.FileMode", mode_class_mock), patch(
with patch("darker.formatters.black_wrapper.FileMode", mode_class_mock), patch(
"black.format_str", format_str
):
# end of test setup, now call the function under test
Expand Down
8 changes: 4 additions & 4 deletions src/darker/tests/test_formatters_black.py
Original file line number Diff line number Diff line change
Expand Up @@ -278,7 +278,7 @@ def test_run(encoding, newline):
def test_run_always_uses_unix_newlines(newline):
"""Content is always passed to Black with Unix newlines"""
src = TextDocument.from_str(f"print ( 'touché' ){newline}")
with patch("black.format_str") as format_str:
with patch("darker.formatters.black_wrapper.format_str") as format_str:
format_str.return_value = 'print("touché")\n'

_ = BlackFormatter().run(src)
Expand Down Expand Up @@ -390,9 +390,9 @@ def test_run_configuration(
):
"""`BlackFormatter.run` passes correct configuration to Black."""
src = TextDocument.from_str("import os\n")
with patch("black.format_str") as format_str, raises_or_matches(
expect, []
) as check:
with patch(
"darker.formatters.black_wrapper.format_str"
) as format_str, raises_or_matches(expect, []) as check:
format_str.return_value = "import os\n"
formatter = BlackFormatter()
formatter.config = black_config
Expand Down

0 comments on commit e8ca107

Please sign in to comment.