Skip to content

Commit

Permalink
fix: pyupgrade entire modified files, not only user-modified chunks
Browse files Browse the repository at this point in the history
This is a limitation of Darker+Pyupgrade.
  • Loading branch information
akaihola committed Dec 31, 2024
1 parent ce8f10c commit bdba037
Show file tree
Hide file tree
Showing 6 changed files with 32 additions and 25 deletions.
52 changes: 27 additions & 25 deletions src/darker/__main__.py
Original file line number Diff line number Diff line change
Expand Up @@ -227,9 +227,17 @@ def _reformat_and_flynt_single_file( # noqa: PLR0913
"no" if formatted == fstringified else "some",
len(formatted.lines),
)
# 4. apply all re-formatter modifications if the re-formatter doesn't guarantee
# preserving the abstract syntax tree (AST); otherwise do steps 5 to 10
if not formatter.preserves_ast:
logger.debug(
"Preserving the AST not guaranteed by %s, applying all changes",
formatter.name,
)
return formatted

# 4. get a diff between the edited to-file and the processed content
# 5. convert the diff into chunks, keeping original and reformatted content for each
# 5. get a diff between the edited to-file and the processed content
# 6. convert the diff into chunks, keeping original and reformatted content for each
# chunk
new_chunks = diff_chunks(rev2_isorted, formatted)

Expand Down Expand Up @@ -337,9 +345,9 @@ def _drop_changes_on_unedited_lines(
context_lines,
abspath_in_rev2,
)
# 6. diff the given revisions (optionally with isort modifications) for each
# 7. diff the given revisions (optionally with isort modifications) for each
# file
# 7. extract line numbers in each edited to-file for changed lines
# 8. extract line numbers in each edited to-file for changed lines
edited_linenums = edited_linenums_differ.revision_vs_lines(
relpath_in_repo, rev2_isorted, context_lines
)
Expand All @@ -348,7 +356,7 @@ def _drop_changes_on_unedited_lines(
last_successful_reformat = rev2_isorted
break

# 8. choose processed content for each chunk if there were any changed lines
# 9. choose processed content for each chunk if there were any changed lines
# inside the chunk in the edited to-file, or choose the chunk's original
# contents if no edits were done in that chunk
chosen = TextDocument.from_lines(
Expand All @@ -358,8 +366,8 @@ def _drop_changes_on_unedited_lines(
mtime=datetime.utcnow().strftime(GIT_DATEFORMAT),
)

# 9. verify that the resulting reformatted source code parses to an identical
# AST as the original edited to-file
# 10. verify that the resulting reformatted source code parses to an identical
# AST as the original edited to-file
if not has_fstring_changes and not verifier.is_equivalent_to_baseline(chosen):
logger.debug(
"Verifying that the %s original edited lines and %s reformatted lines "
Expand Down Expand Up @@ -459,32 +467,26 @@ def _import_pygments(): # type: ignore
def main( # noqa: C901,PLR0912,PLR0915
argv: List[str] = None,
) -> int:
"""Parse the command line and reformat and optionally lint each source file
"""Parse the command line and reformat and optionally lint each source file.
1. run isort on each edited file (optional)
2. run flynt (optional) on the isorted contents of each edited to-file
3. run a code re-formatter on the isorted and fstringified contents of each edited
to-file
4. get a diff between the edited to-file and the processed content
5. convert the diff into chunks, keeping original and reformatted content for each
4. apply all re-formatter modifications if the re-formatter doesn't guarantee
preserving the abstract syntax tree (AST); otherwise do steps 5 to 10
5. get a diff between the edited to-file and the processed content
6. convert the diff into chunks, keeping original and reformatted content for each
chunk
6. diff the given revisions (optionally with isort modifications) for each
7. diff the given revisions (optionally with isort modifications) for each
file
7. extract line numbers in each edited to-file for changed lines
8. choose processed content for each chunk if there were any changed lines inside
8. extract line numbers in each edited to-file for changed lines
9. choose processed content for each chunk if there were any changed lines inside
the chunk in the edited to-file, or choose the chunk's original contents if no
edits were done in that chunk
9. verify that the resulting reformatted source code parses to an identical AST as
the original edited to-file
9. write the reformatted source back to the original file or print the diff
10. run linter subprocesses twice for all modified and unmodified files which are
mentioned on the command line: first establish a baseline by running against
``rev1``, then get current linting status by running against the working tree
(steps 10.-12. are optional)
11. create a mapping from line numbers of unmodified lines in the current versions
to corresponding line numbers in ``rev1``
12. hide linter messages which appear in the current versions and identically on
corresponding lines in ``rev1``, and show all other linter messages
10. verify that the resulting reformatted source code parses to an identical AST as
the original edited to-file
11. write the reformatted source back to the original file or print the diff
:param argv: The command line arguments to the ``darker`` command
:return: 1 if the ``--check`` argument was provided and at least one file was (or
Expand Down Expand Up @@ -611,7 +613,7 @@ def main( # noqa: C901,PLR0912,PLR0915
workers=config["workers"],
),
):
# 10. A re-formatted Python file which produces an identical AST was
# 11. A re-formatted Python file which produces an identical AST was
# created successfully - write an updated file or print the diff if
# there were any changes to the original
formatting_failures_on_modified_lines = True
Expand Down
1 change: 1 addition & 0 deletions src/darker/formatters/base_formatter.py
Original file line number Diff line number Diff line change
Expand Up @@ -29,6 +29,7 @@ class BaseFormatter(HasConfig[FormatterConfig]):
"""Base class for code re-formatters."""

name: str
preserves_ast: bool

def read_config(self, src: tuple[str, ...], args: Namespace) -> None:
"""Read code re-formatter configuration from a configuration file.
Expand Down
1 change: 1 addition & 0 deletions src/darker/formatters/black_formatter.py
Original file line number Diff line number Diff line change
Expand Up @@ -80,6 +80,7 @@ class BlackFormatter(BaseFormatter, HasConfig[BlackCompatibleConfig]):

name = "black"
config_section = "tool.black"
preserves_ast = True

def read_config(self, src: tuple[str, ...], args: Namespace) -> None:
"""Read Black configuration from ``pyproject.toml``.
Expand Down
1 change: 1 addition & 0 deletions src/darker/formatters/none_formatter.py
Original file line number Diff line number Diff line change
Expand Up @@ -17,6 +17,7 @@ class NoneFormatter(BaseFormatter):
"""A dummy code formatter plugin interface."""

name = "dummy reformat"
preserves_ast = True

def run(
self, content: TextDocument, path_from_cwd: Path
Expand Down
1 change: 1 addition & 0 deletions src/darker/formatters/pyupgrade_formatter.py
Original file line number Diff line number Diff line change
Expand Up @@ -60,6 +60,7 @@ class PyupgradeFormatter(BaseFormatter, HasConfig[PyupgradeConfig]):
config: PyupgradeConfig # type: ignore[assignment]

name = "pyupgrade"
preserves_ast = False

def run(
self, content: TextDocument, path_from_cwd: Path
Expand Down
1 change: 1 addition & 0 deletions src/darker/formatters/ruff_formatter.py
Original file line number Diff line number Diff line change
Expand Up @@ -76,6 +76,7 @@ class RuffFormatter(BaseFormatter, HasConfig[BlackCompatibleConfig]):

name = "ruff format"
config_section = "tool.ruff"
preserves_ast = True

def run(self, content: TextDocument, path_from_cwd: Path) -> TextDocument:
"""Run the Ruff code re-formatter for the Python source code given as a string.
Expand Down

0 comments on commit bdba037

Please sign in to comment.