-
Notifications
You must be signed in to change notification settings - Fork 1.6k
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
Add suspicious_operation_groupings lint #6086
Add suspicious_operation_groupings lint #6086
Conversation
(rust_highfive has picked a reviewer for you, use r? to override) |
So at the moment several portions of the lint code are incomplete and only compile via the The main reason I'm submitting this as a draft PR intiialy is that as part of this lint I would like to provide a suggestion that swaps a dynamically discovered identifier in an expression with another identifier. Creating a suggestion by changing an identifier in an expression seems like the kind of things that might have already been needed elsewhere. But after searching for a while in the clippy codebase I wasn't able to find a previous example of that. Before attempting to create a way to do that, I'd like to know whether I've missed someplace where identifier swapping, or something similar has already been done. Alternatively, I'd also be interested in ideas anyone has on how to implement identifier swapping, given it has not been implemented already. My best guess at how to implement it would be to recurse down the AST and find the |
After taking the time to do a more thorough search of the clippy code base, I have concluded that the most similar thing to what I need here that already exists is |
9a1e611
to
b85ac8d
Compare
☔ The latest upstream changes (presumably #6117) made this pull request unmergeable. Please resolve the merge conflicts. Note that reviewers usually do not review pull requests until merge conflicts are resolved! Once you resolve the conflicts, you should change the labels applied by bors to indicate that your PR is ready for review. Post this as a comment to change the labels:
|
442bb91
to
8658110
Compare
Hey @Ryan1729, I've seen there is another WIP PR that this one depends on, and that you plan to rework that PR using a visitor. Feel free to ping me when you think this one is ready for review :) |
bfcae7b
to
09eacac
Compare
error: This sequence of operators looks suspiciously like a bug. | ||
--> $DIR/suspicious_operation_groupings.rs:11:9 | ||
| | ||
LL | self.x == other.y && self.y == other.y && self.z == other.z |
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.
This lint message is emitted twice, and my best guess as to why, is that both self.x == other.y && self.y == other.y && self.z == other.z
and self.x == other.y && self.y == other.y
have detectable errors, and they both get visited separately by the lint pass.
I'm unaware of a good way to prevent these double messages. Is this something that has been addressed for other lints before?
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.
I can't remember now a similar case. I think you can leave it like this for now and once we get to review we can try to find a solution.
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.
I think that having a double message is acceptable because the lint will likely find a lot of real bugs, so we can look into this as a follow-up PR.
With that said, I believe the correct solution would be to look at the parent Expr
if any and collect all exprs
that trigger the lint into one multi-suggestion.
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.
How do we prevent the sub-expressions from triggering the lint again after the parent expression collects all the suggestions after that? As far as I can tell check_expr
gets called for each expr
and each expr
contained inside that expr
.
4bcb20b
to
91f4063
Compare
@ebroto As far as I understand the current CI failure is due to a change in |
r? @llogiq I'm a bit overwhelmed by PRs and quite short on time, would you mind taking over this one? Feel free to reassign it to me if you can't! I think once this is ready we should remember to squash as there are many intermediate commits. |
Sure thing, @ebroto! |
Yeah, I can tell that this PR has gotten large. That’s why I decided to postpone a feature and make issue #6275 about getting to that later. Would it be useful to squash commits containing consecutive rewrites of the same parts of the code, now, before review? Is there anything I can do to make reviewing this PR easier? Splitting the PR into smaller PRs maybe? |
No, it's all good, I'll just need more time. Give me about two weeks. |
2d138fb
to
2e05439
Compare
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.
First round of review, no blockers so far but a few suggestions.
error: This sequence of operators looks suspiciously like a bug. | ||
--> $DIR/suspicious_operation_groupings.rs:11:9 | ||
| | ||
LL | self.x == other.y && self.y == other.y && self.z == other.z |
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.
I think that having a double message is acceptable because the lint will likely find a lot of real bugs, so we can look into this as a follow-up PR.
With that said, I believe the correct solution would be to look at the parent Expr
if any and collect all exprs
that trigger the lint into one multi-suggestion.
/// } | ||
/// ``` | ||
pub SUSPICIOUS_OPERATION_GROUPINGS, | ||
correctness, |
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.
I feel that correctness
has a higher bar than flagging suspicious operations. Correctness lints are denied, not warned, and I feel that'd be a bad idea. On the other hand, I am not sure where to place the lint. Neither complexity
nor style
seem an obvious solution (although I'd lean towards style
if we want to make it a warn
by default lint), and perf
is obviously the wrong choice. That would leave pedantic
or restriction
, so when choosing between the two, I would choose pedantic
.
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.
suspicious_arithmetic_impl
is a complexity
lint, but I guess that triggers on a much smaller set of possible code. If it turned out that, in practice, the accuracy of this lint was sufficiently high, (that is, if no one ever actually needed to do the operations this lint flags,) then I would argue that correctness
would still make sense. But admittedly, we don't know how accurate this lint it in practice.
The unfortunate thing is that if the lint doesn't at least warn by default, then fewer people will end up using it, and we will have less data about how often people really want to do operations that this lint flags as suspicious. I think style
is also a good choice since if the lint does turn out to have false positives, then (I assume) people would be more comfortable disabling it.
So given all that I'd rather change it to style
, but if there is strong opposition to that, then changing it to pedantic
makes sense to me too.
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.
Ok, then let's make it a style
lint for now. We really need another category for lints like this.
☔ The latest upstream changes (presumably #6134) made this pull request unmergeable. Please resolve the merge conflicts. Note that reviewers usually do not review pull requests until merge conflicts are resolved! Once you resolve the conflicts, you should change the labels applied by bors to indicate that your PR is ready for review. Post this as a comment to change the labels:
|
e6342d0
to
6d444a2
Compare
r=me with lint group changed to |
ee6e149
to
89a9c27
Compare
@llogiq I've squashed some of the commits as @ebroto suggested above, and the pushed commits now include making the lint a style lint. |
I think @ebroto meant to squash all the commits. Otherwise great job! 👍 |
Hmm. If I look at the recent history on this repo I see series of multiple commits that seem to come from the same PR. @ebroto , can you clarify what you wanted? I was under the impression that we only wanted to squash commits containing code that was replaced later by other code. |
☔ The latest upstream changes (presumably #6333) made this pull request unmergeable. Please resolve the merge conflicts. Note that reviewers usually do not review pull requests until merge conflicts are resolved! Once you resolve the conflicts, you should change the labels applied by bors to indicate that your PR is ready for review. Post this as a comment to change the labels:
|
abae85d
to
219a71f
Compare
☔ The latest upstream changes (presumably #6389) made this pull request unmergeable. Please resolve the merge conflicts. Note that reviewers usually do not review pull requests until merge conflicts are resolved! Once you resolve the conflicts, you should change the labels applied by bors to indicate that your PR is ready for review. Post this as a comment to change the labels:
|
run `cargo dev new_lint --category correctness --name suspicious_chained_operators --pass early` add (currently failing) tests for suspicious_chained_operators add some tests to answer a question that came up during implementation write usage code for functions we'll need to find or create Complete left-right tracking TODO get it compiling with several `todo!` invocations. refactor to a set of incomplete functions that don't expect to be able to edit a `Span` create placeholder for `suggestion_with_swapped_ident` function and correct some comments add `inside_larger_boolean_expression` test fill out `get_ident` and `suggestion_with_swapped_ident` Implementi the `IdentIter` start on implementing the `IdentIter` handle the `ExprKind::Path` case in `IdentIter` on second thought, make the iterator type dynamic so we don't need an explicit type for each one we will need handle `ExprKind::MacCall` in `IdentIter` Try handling `box x` expressions restructure `IdentIter` set `self.done` when returning `None` Handle `ExprKind::Array` reduce duplication with a macro that we expect to use several more times handle ExprKind::Call add `new_p` convenience method handle `MethodCall` handle `Tup` and `Binary` handle `Unary` simplify by not returning an additional `Expr` from the `IdentIter` add cross product test against false positives rename suspicious_chained_operators to suspicious_operation_groupings within files For the record, the exact commands run were: find . -type f -name "*.md" -exec sed -i 's/suspicious_chained_operators/suspicious_operation_groupings/g' {} + find . -type f -name "*.rs" -exec sed -i 's/suspicious_chained_operators/suspicious_operation_groupings/g' {} + find . -type f -name "*.rs" -exec sed -i 's/SUSPICIOUS_CHAINED_OPERATORS/SUSPICIOUS_OPERATION_GROUPINGS/g' {} + find . -type f -name "*.rs" -exec sed -i 's/SuspiciousChainedOperators/SuspiciousOperationGroupings/g' {} + Also: rename file to match module name rename test file to match lint name start implementing `IdentDifference` creation add `IdentIter` utility use `ident_iter::IdentIter` fix bug in `suggestion_with_swapped_ident` add `inside_if_statements` test implement `Add` `todo`s register `SuspiciousOperationGroupings` lint pass fill in `chained_binops`, and fill in a stopgap version of `ident_difference_expr`, but then notice that the lint does not seem to ever be run in the tests run `cargo dev update_lints` and not that the `suspicious_operation_groupings` lint still does not seem to be run fix base index incrementing bug fix paired_identifiers bug, and remove ident from `Single` change help prefix and note our first successful lint messages! add odd_number_of_pairs test get the `non_boolean_operators` test passing, with two copies of the error message extract `is_useless_with_eq_exprs` so we can know when `eq_op` will already handle something add `not_caught_by_eq_op` tests since `s1.b * s1.b` was (reasonably) not caught by `eq_op` cover the case where the change should be made on either side of the expression with `not_caught_by_eq_op` tests produce the expected suggestion on the `not_caught_by_eq_op_middle_change_left` test confirm that the previous tests still pass and update references fix early continue bug and get `not_caught_by_eq_op_middle_change_right` passing note that `not_caught_by_eq_op_start` already passes fix bugs based on misunderstanding of what `Iterator::skip` does, and note that `not_caught_by_eq_op_end` now passes add several parens tests and make some of them pass handle parens inside `chained_binops_helper` and note that this makes several tests pass get `inside_larger_boolean_expression_with_unsorted_ops` test passing by extracting out `check_same_op_binops` function also run `cargo dev fmt` note that `inside_function_call` already passes add another `if_statement` test remove the matching op requirement, making `inside_larger_boolean_expression_with_unsorted_ops` pass prevent non-change suggestions from being emitted get the `Nested` tests passing, and remove apparently false note about eq_op add a test to justify comment in `ident_difference_expr_with_base_location` but find that the failure mode seems different than expected complete `todo` making `do_not_give_bad_suggestions_for_this_unusual_expr` pass and add some more tests that already pass add test to `eq_op` note that `inside_fn_with_similar_expression` already passes fix `inside_an_if_statement` and note that it already passes attempt to implement if statement extraction and notice that we don't seem to handle unary ops correctly add `maximum_unary_minus_right_tree` test and make it pass add two tests and note one of them passes filter out unary operations in several places, and find that the issue seems to be that we don't currently recognize the error in `multiple_comparison_types_and_unary_minus` even so. remove filtering that was causing bad suggestions remove tests that were deemed too much for now run `cargo dev fmt` correct eq_op post-merge fill out the description and delete debugging code run `cargo dev update_lints` update eq_op references add parens to work around rustfmt issue rust-lang#3666 and run rustfmt rust-lang/rustfmt#3666 (comment) update references after formatting fix dogfood issues fix multi-cursor edit fix missed dogfood error fix more dogfood pedantic issues, including function length even more nesting insert hidden definition of Vec3 so docs compile add spaces to second struct def reword test description comment Co-authored-by: llogiq <[email protected]> add local `use BinOpKind::*;` Apply suggestions from code review Co-authored-by: llogiq <[email protected]> switch `SUSPICIOUS_OPERATION_GROUPINGS` to a style lint run `cargo dev update_lints` put both usages of `op_types` in the same closure to satisfy `borrowck` fix compile error
219a71f
to
af1cc5c
Compare
@rustbot modify labels: +S-waiting-on-review -S-waiting-on-author |
Sorry for the delays. Thank you for seeing this through with us. @bors r+ |
📌 Commit af1cc5c has been approved by |
☀️ Test successful - checks-action_dev_test, checks-action_remark_test, checks-action_test |
This is my (
currently WIP) attempt to close #6039.changelog: Added
suspicious_operation_groupings
lint.