-
Notifications
You must be signed in to change notification settings - Fork 2.5k
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Fix bug where # fmt: skip is not being respected with one-liner functions #4552
base: main
Are you sure you want to change the base?
Conversation
a94a78c
to
2433e85
Compare
I’d also like to mention that I included this in the Stable Style, as I considered it more of a bug fix than a new feature. However, I’m not yet fully familiar with the criteria for what qualifies as Stable versus Preview Style, and I couldn’t find much information about this in the documentation. Any guidance would be greatly appreciated. |
tests/data/cases/fmtskip10.py
Outdated
@@ -0,0 +1 @@ | |||
def foo(): return "mock" # fmt: skip |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
def foo(): return "mock" # fmt: skip | |
def foo(): return "mock" # fmt: skip | |
if True: print("yay") # fmt: skip | |
for i in range(10): print(i) # fmt: skip |
Looks to also fix #3682 so this adds two examples from that as test cases. The other issues not fixed by this PR in #3682 look to have been fixed by something in the 03-15-2024 release, probably #4248, so it should be good to add to the close-on-merge list for this PR along with #4535
(To do that, edit the opening description to include
Fixes #3682
Fixes #4535
)
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Thanks for the review.
I added the suggested tests + some extra ones.
I also mentioned #3682 on the PR description as suggested.
Typically speaking, everything should go in the preview style, the only exceptions being bug fixes for crashes and code changes that don't change code that has already been formatted by Black. This fits under the second case, so it does not need to go through the preview style if it doesn't make sense for it to. |
2433e85
to
c37f259
Compare
Thank you for the clarification! I’ve already updated my PR to use the preview style. During this process, the test I added naturally started failing, since it was running on stable mode. I attempted to give the --preview flag to the test case, but it didn’t resolve the issue. The test only worked when I gave it the Is this the correct approach? If so, could you clarify the intended purpose of the |
The idea of the stable style is that if you have a codebase formatted with Black (say) 25.1, and you upgrade to Black 25.2, nothing should change. This is useful so that users can upgrade Black and see bugfixes without having to commit reformatting. If this change only affects how Black formats code that wasn't previously formatted by Black. Putting |
src/black/comments.py
Outdated
siblings.insert(0, prev_sibling) | ||
yield from siblings | ||
|
||
if Preview.multiline_string_handling not in mode: |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Wrong preview feature?
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Yep, you are absolutely right.
Sorry for missing this. This was the reason for the tests failing in --preview
mode (the multiline_string_handling
is an unstable feature).
c37f259
to
ee282ae
Compare
Description
Fixes #3682
Fixes #4535
The fix involves modifying the algorithm that determines which nodes to ignore in the
_generate_ignored_nodes_from_fmt_skip
function. Specifically, the adjustment applies to cases where the# fmt: skip
comment is not immediately after the colon in anif
,while
,def
,class
, etc.Previous Algorithm
The previous algorithm for selecting ignored nodes worked as follows:
\n
in its prefix.This approach worked well when the node with the comment existed at a tree level that covered the entire line. However, it failed in cases where this assumption was invalid. For example, consider the tree produced for the code in #4535:
As shown, only the content of the
return "mock" # fmt: skip
would be ignored.This can be tested with the following input (note the single quotes):
The resulting output would be:
Here, the single quotes are protected by the skip statement.
New Algorithm
The new algorithm traverses the tree differently, ensuring it can visit all leaves in the order they appear in the source code (but in reverse). For the example tree:
:
), then to theRPAR
,LPAR
, and so on.NEWLINE
orINDENT
node.# fmt: skip
, expect theNEWLINE
orINDENT
node.This approach ensures all relevant nodes are included in the ignored list.
Checklist - did you ...
CHANGES.md
if necessary?