Skip to content

Commit

Permalink
feat: update to black v24
Browse files Browse the repository at this point in the history
  • Loading branch information
mbhall88 committed Jan 29, 2024
1 parent ecc1907 commit 8ee74cd
Show file tree
Hide file tree
Showing 11 changed files with 390 additions and 294 deletions.
556 changes: 278 additions & 278 deletions poetry.lock

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ snakefmt = 'snakefmt.snakefmt:main'
[tool.poetry.dependencies]
python = "^3.8.1"
click = "^8.0.0"
black = "^23.12.1"
black = "^24.1.1"
toml = "^0.10.2"
importlib_metadata = {version = ">=1.7.0,<5.0", python = "<3.8"}

Expand Down
9 changes: 9 additions & 0 deletions snakefmt/__init__.py
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,8 @@
(in dist-info or egg-info dirs).
From Python 3.8, importlib_metadata is in standard library as importlib.metadata.
"""
from black import TargetVersion

if sys.version_info >= (3, 8):
from importlib import metadata
else:
Expand All @@ -14,3 +16,10 @@
__version__ = metadata.version("snakefmt")

DEFAULT_LINE_LENGTH = 88
DEFAULT_TARGET_VERSIONS = {
TargetVersion.PY38,
TargetVersion.PY39,
TargetVersion.PY310,
TargetVersion.PY311,
TargetVersion.PY312,
}
6 changes: 4 additions & 2 deletions snakefmt/config.py
Original file line number Diff line number Diff line change
Expand Up @@ -10,7 +10,7 @@
import toml
from black import Mode, find_project_root

from snakefmt import DEFAULT_LINE_LENGTH
from snakefmt import DEFAULT_LINE_LENGTH, DEFAULT_TARGET_VERSIONS
from snakefmt.exceptions import MalformattedToml

PathLike = Union[Path, str]
Expand Down Expand Up @@ -57,7 +57,9 @@ def inject_snakefmt_config(

def read_black_config(path: Optional[PathLike]) -> Mode:
"""Parse Black configuration from provided toml."""
black_mode = Mode(line_length=DEFAULT_LINE_LENGTH)
black_mode = Mode(
line_length=DEFAULT_LINE_LENGTH, target_versions=DEFAULT_TARGET_VERSIONS
)
if path is None:
return black_mode
if not Path(path).is_file():
Expand Down
40 changes: 40 additions & 0 deletions snakefmt/formatter.py
Original file line number Diff line number Diff line change
Expand Up @@ -26,6 +26,10 @@
full_string_matcher = re.compile(
r"^\s*(\w?([\"']{3}.*?[\"']{3})|([\"']{1}.*?[\"']{1}))$", re.DOTALL | re.MULTILINE
)
# this regex matches any docstring; can span multiple lines
docstring_matcher = re.compile(
r"\s*([rR]?[\"']{3}.*?[\"']{3})", re.DOTALL | re.MULTILINE
)
contextual_matcher = re.compile(
r"(.*)^(if|elif|else|with|for|while)([^:]*)(:.*)", re.S | re.M
)
Expand All @@ -41,6 +45,17 @@ def is_all_comments(string):
)


def index_of_first_docstring(s: str) -> Optional[int]:
"""
Returns the index (i.e., index of last quote character) of the first docstring in
a string, or None if there are no docstrings.
"""
match = docstring_matcher.search(s)
if match is None:
return None
return match.end(1) - 1


class Formatter(Parser):
def __init__(
self,
Expand Down Expand Up @@ -296,9 +311,34 @@ def format_param(
if param_list:
val = f"f({val})"
extra_spacing = 3

# get the index of the last character of the first docstring, if any
docstring_index = index_of_first_docstring(val)
docstring_line_index = None
if docstring_index is not None:
docstring_line_index = val[:docstring_index].count("\n")
lines = val.splitlines()
if docstring_line_index is not None and docstring_line_index + 1 < len(lines):
docstring_has_extra_newline_after = (
lines[docstring_line_index + 1].strip() == ""
)
else:
docstring_has_extra_newline_after = False

val = self.run_black_format_str(
val, target_indent, extra_spacing, no_nesting=True
)

# remove newline added after first docstring (black>=24.1)
if docstring_line_index is not None and not docstring_has_extra_newline_after:
lines = val.splitlines()
if docstring_line_index + 1 < len(lines):
line_after_docstring = lines[docstring_line_index + 1]
if line_after_docstring.strip() == "":
# delete the newline
lines.pop(docstring_line_index + 1)
val = "\n".join(lines)

if param_list:
match_equal = re.match(r"f\((.*)\)", val, re.DOTALL)
val = match_equal.group(1)
Expand Down
1 change: 1 addition & 0 deletions snakefmt/parser/syntax.py
Original file line number Diff line number Diff line change
@@ -1,6 +1,7 @@
"""
Code in charge of parsing and validating Snakemake syntax
"""

import sys
import tokenize
from abc import ABC, abstractmethod
Expand Down
4 changes: 3 additions & 1 deletion snakefmt/snakefmt.py
Original file line number Diff line number Diff line change
Expand Up @@ -288,7 +288,9 @@ def main(

if check:
if files_unchanged == len(files_to_format):
logger.info(f"All {len(files_to_format)} file(s) would be left unchanged 🎉")
logger.info(
f"All {len(files_to_format)} file(s) would be left unchanged 🎉"
)
ctx.exit(ExitCode.NO_CHANGE.value)
elif files_with_errors > 0:
exit_value = ExitCode.ERROR.value
Expand Down
36 changes: 27 additions & 9 deletions tests/test_config.py
Original file line number Diff line number Diff line change
Expand Up @@ -5,7 +5,7 @@
import click
import pytest

from snakefmt import DEFAULT_LINE_LENGTH
from snakefmt import DEFAULT_LINE_LENGTH, DEFAULT_TARGET_VERSIONS
from snakefmt.config import (
find_pyproject_toml,
inject_snakefmt_config,
Expand Down Expand Up @@ -191,7 +191,9 @@ def test_empty_config_default_line_length_used(self, tmp_path):
formatter = setup_formatter("")
path = tmp_path / "config.toml"
path.touch()
expected = black.FileMode(line_length=DEFAULT_LINE_LENGTH)
expected = black.FileMode(
line_length=DEFAULT_LINE_LENGTH, target_versions=DEFAULT_TARGET_VERSIONS
)
assert formatter.black_mode == expected

def test_read_black_config_settings(self, tmp_path):
Expand All @@ -200,7 +202,9 @@ def test_read_black_config_settings(self, tmp_path):
path.write_text(f"[tool.black]\nline_length = {black_line_length}")

actual = read_black_config(path)
expected = black.FileMode(line_length=black_line_length)
expected = black.FileMode(
line_length=black_line_length, target_versions=DEFAULT_TARGET_VERSIONS
)

assert actual == expected

Expand All @@ -213,14 +217,18 @@ def test_snakefmt_line_length_overrides_black(self, tmp_path):
# show black gets parsed
formatter = setup_formatter("", black_config_file=str(path))

expected = black.FileMode(line_length=black_line_length)
expected = black.FileMode(
line_length=black_line_length, target_versions=DEFAULT_TARGET_VERSIONS
)
assert formatter.black_mode == expected

# Now, add overriding snakefmt line length
formatter = setup_formatter(
"", line_length=snakefmt_line_length, black_config_file=str(path)
)
expected = black.FileMode(line_length=snakefmt_line_length)
expected = black.FileMode(
line_length=snakefmt_line_length, target_versions=DEFAULT_TARGET_VERSIONS
)
assert formatter.black_mode == expected

def test_unrecognised_black_options_in_config_ignored_and_default_line_length_used(
Expand All @@ -232,7 +240,9 @@ def test_unrecognised_black_options_in_config_ignored_and_default_line_length_us

read_black_config(path)
actual = formatter.black_mode
expected = black.FileMode(line_length=DEFAULT_LINE_LENGTH)
expected = black.FileMode(
line_length=DEFAULT_LINE_LENGTH, target_versions=DEFAULT_TARGET_VERSIONS
)

assert actual == expected

Expand All @@ -253,7 +263,9 @@ def test_skip_string_normalisation_handled_with_snakecase(self, tmp_path):
read_black_config(path)
actual = formatter.black_mode
expected = black.FileMode(
line_length=DEFAULT_LINE_LENGTH, string_normalization=True
line_length=DEFAULT_LINE_LENGTH,
string_normalization=True,
target_versions=DEFAULT_TARGET_VERSIONS,
)

assert actual == expected
Expand All @@ -266,7 +278,9 @@ def test_skip_string_normalisation_handled_with_kebabcase(self, tmp_path):
read_black_config(path)
actual = formatter.black_mode
expected = black.FileMode(
line_length=DEFAULT_LINE_LENGTH, string_normalization=True
line_length=DEFAULT_LINE_LENGTH,
string_normalization=True,
target_versions=DEFAULT_TARGET_VERSIONS,
)

assert actual == expected
Expand All @@ -279,5 +293,9 @@ def test_string_normalisation_handled(self, tmp_path):
"", line_length=line_length, black_config_file=str(path)
)

expected = black.FileMode(line_length=line_length, string_normalization=False)
expected = black.FileMode(
line_length=line_length,
string_normalization=False,
target_versions=DEFAULT_TARGET_VERSIONS,
)
assert formatter.black_mode == expected
28 changes: 25 additions & 3 deletions tests/test_formatter.py
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@
The tests implicitly assume that the input syntax is correct ie that no parsing-related
errors arise, as tested in test_parser.py.
"""

from io import StringIO
from unittest import mock

Expand Down Expand Up @@ -787,7 +788,6 @@ def test_single_quoted_multiline_string_proper_tabbing(self):
assert formatter.get_formatted() == expected

def test_docstrings_get_retabbed_for_snakecode_only(self):
"""Black only retabs the first tpq in a docstring."""
snakecode = '''def f():
"""Does not do
much
Expand All @@ -804,7 +804,8 @@ def test_docstrings_get_retabbed_for_snakecode_only(self):
formatter = setup_formatter(snakecode)
expected = f'''def f():
{TAB * 1}"""Does not do
much"""
{TAB * 1}much
{TAB * 1}"""
{TAB * 1}pass
Expand Down Expand Up @@ -1431,4 +1432,25 @@ def test_shell_indention_long_line(self):
f"{TAB * 2})\n"
)
formatter = setup_formatter(snakecode)
assert formatter.get_formatted() == snakecode

expected = (
"rule test1:\n"
f"{TAB * 1}input:\n"
f'{TAB * 2}"...",\n'
f"{TAB * 1}output:\n"
f'{TAB * 2}"...",\n'
f"{TAB * 1}shell:\n"
f"{TAB * 2}myfunc(\n"
f'{TAB * 3}"param1",\n'
f"{TAB * 3}[\n"
f'{TAB * 4}"item1",\n'
f"{TAB * 4}(\n"
f'{TAB * 5}f"very_long_item2_{{very_long_function(other_param)}}"\n'
f"{TAB * 5}if some_very_long_condition\n"
f'{TAB * 5}else ""\n'
f"{TAB * 4}),\n"
f"{TAB * 3}],\n"
f"{TAB * 2})\n"
)

assert formatter.get_formatted() == expected
1 change: 1 addition & 0 deletions tests/test_grammar.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@
Completeness tests: checks that the grammar used is a bijection of the snakemake grammar
To use the latest snakemake grammar, run `poetry update snakemake` from this repo
"""

from snakemake import parser

from snakefmt.parser import grammar
Expand Down
1 change: 1 addition & 0 deletions tests/test_parser.py
Original file line number Diff line number Diff line change
Expand Up @@ -2,6 +2,7 @@
Examples where we raise errors but snakemake does not are listed as 'SMK_NOBREAK'
"""

from io import StringIO

import pytest
Expand Down

0 comments on commit 8ee74cd

Please sign in to comment.