Skip to content
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

Issue 3267: fixed wps226 location #3269

Merged
merged 2 commits into from
Feb 6, 2025
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
2 changes: 1 addition & 1 deletion CHANGELOG.md
Original file line number Diff line number Diff line change
Expand Up @@ -29,9 +29,9 @@ This version introduces `wps` CLI tool.
- Fixes `WPS115` false-positive on `Enum` attributes, #3238
- Removes duplicated `WPS312`, #3239
- Fixes `WPS432`, now it shows literal num, #1402
- Fixes `WPS226`, now it points to the first string literal occurrence, #3267
- Fixes `WPS605` false-positive on `@staticmethod`, #3292


## 1.0.0

### Ruff
Expand Down
15 changes: 15 additions & 0 deletions tests/test_visitors/conftest.py
Original file line number Diff line number Diff line change
Expand Up @@ -100,3 +100,18 @@ def factory(
assert reproduction.message() == violation.message()

return factory


@pytest.fixture(scope='session')
def assert_error_location():
"""Helper function to assert visitor violation location is expected."""

def factory(
visitor: BaseVisitor,
expected: tuple[int, int],
) -> None:
assert len(visitor.violations) == 1
violation = visitor.violations[0]
assert violation._location() == expected # noqa: SLF001

return factory
Original file line number Diff line number Diff line change
Expand Up @@ -96,6 +96,8 @@ def function() -> Dict[int, {0}]:
...
"""

EXPECTED_LOCATION = (2, 8)


@pytest.mark.parametrize(
'strings',
Expand Down Expand Up @@ -159,6 +161,7 @@ def test_string_overuse_settings(
def test_string_overuse(
assert_errors,
assert_error_text,
assert_error_location,
parse_ast_tree,
default_options,
strings,
Expand All @@ -167,7 +170,6 @@ def test_string_overuse(
):
"""Ensures that over-used strings raise violations."""
tree = parse_ast_tree(strings.format(prefix + string_value))

visitor = StringOveruseVisitor(default_options, tree=tree)
visitor.run()

Expand All @@ -177,6 +179,7 @@ def test_string_overuse(
string_value.replace('"', '') or "''",
default_options.max_string_usages,
)
assert_error_location(visitor, EXPECTED_LOCATION)


@pytest.mark.parametrize(
Expand Down
2 changes: 2 additions & 0 deletions wemake_python_styleguide/violations/complexity.py
Original file line number Diff line number Diff line change
Expand Up @@ -881,6 +881,8 @@ class OverusedStringViolation(MaybeASTViolation):
single space `' '`, new line `'\n'`, `'\r\n'` and tabulator `'\t'`
do not count against string literal overuse limit.

The violation points to the first occurrence of the overused string literal.

Reasoning:
When some string is used more than several time in your code,
it probably means that this string is a meaningful constant
Expand Down
9 changes: 9 additions & 0 deletions wemake_python_styleguide/visitors/ast/complexity/overuses.py
Original file line number Diff line number Diff line change
Expand Up @@ -67,6 +67,11 @@ def __init__(self, *args, **kwargs) -> None:
int,
] = defaultdict(int)

self._string_constants_first_node: defaultdict[
Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

This can be self._string_constants_first_node: ast.AST | None = None

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Explain the idea, please

Copy link
Member

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

We only need the first node, right? We don't need all nodes. So, we can only save the first node here.

Copy link
Contributor Author

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Yes. And it is done so. For each string their first nodes are collected in dict.

AnyTextPrimitive,
ast.Constant,
] = defaultdict(lambda: ast.Constant(value=None))

def visit_any_string(self, node: ast.Constant) -> None:
"""Restricts to over-use string constants."""
self._check_string_constant(node)
Expand All @@ -81,6 +86,9 @@ def _check_string_constant(self, node: ast.Constant) -> None:
if node.value in self._ignored_string_constants:
return

if node.value not in self._string_constants_first_node:
self._string_constants_first_node[node.value] = node

self._string_constants[node.value] += 1

def _post_visit(self) -> None:
Expand All @@ -90,6 +98,7 @@ def _post_visit(self) -> None:
complexity.OverusedStringViolation(
text=source.render_string(string) or "''",
baseline=self.options.max_string_usages,
node=self._string_constants_first_node[string],
),
)

Expand Down