mirror of
https://github.com/astral-sh/ruff.git
synced 2025-09-28 12:55:05 +00:00
Disable auto-fix when source has syntax errors (#12134)
## Summary This PR updates Ruff to **not** generate auto-fixes if the source code contains syntax errors as determined by the parser. The main motivation behind this is to avoid infinite autofix loop when the token-based rules are run over any source with syntax errors in #11950. Although even after this, it's not certain that there won't be an infinite autofix loop because the logic might be incorrect. For example, https://github.com/astral-sh/ruff/issues/12094 and https://github.com/astral-sh/ruff/pull/12136. This requires updating the test infrastructure to not validate for fix availability status when the source contained syntax errors. This is required because otherwise the fuzzer might fail as it uses the test function to run the linter and validate the source code. resolves: #11455 ## Test Plan `cargo insta test`
This commit is contained in:
parent
dcb9523b1e
commit
88a4cc41f7
7 changed files with 180 additions and 155 deletions
|
@ -204,12 +204,12 @@ pub(crate) fn test_contents<'a>(
|
|||
print_syntax_errors(parsed.errors(), path, &locator, &transformed);
|
||||
|
||||
panic!(
|
||||
r#"Fixed source has a syntax error where the source document does not. This is a bug in one of the generated fixes:
|
||||
"Fixed source has a syntax error where the source document does not. This is a bug in one of the generated fixes:
|
||||
{syntax_errors}
|
||||
Last generated fixes:
|
||||
{fixes}
|
||||
Source with applied fixes:
|
||||
{}"#,
|
||||
{}",
|
||||
transformed.source_code()
|
||||
);
|
||||
}
|
||||
|
@ -228,7 +228,12 @@ Source with applied fixes:
|
|||
.into_iter()
|
||||
.map(|diagnostic| {
|
||||
let rule = diagnostic.kind.rule();
|
||||
let fixable = diagnostic.fix.as_ref().is_some_and(|fix| matches!(fix.applicability(), Applicability::Safe | Applicability::Unsafe));
|
||||
let fixable = diagnostic.fix.as_ref().is_some_and(|fix| {
|
||||
matches!(
|
||||
fix.applicability(),
|
||||
Applicability::Safe | Applicability::Unsafe
|
||||
)
|
||||
});
|
||||
|
||||
match (fixable, rule.fixable()) {
|
||||
(true, FixAvailability::Sometimes | FixAvailability::Always)
|
||||
|
@ -236,28 +241,39 @@ Source with applied fixes:
|
|||
// Ok
|
||||
}
|
||||
(true, FixAvailability::None) => {
|
||||
panic!("Rule {rule:?} is marked as non-fixable but it created a fix. Change the `Violation::FIX_AVAILABILITY` to either `FixAvailability::Sometimes` or `FixAvailability::Always`");
|
||||
},
|
||||
panic!(
|
||||
"Rule {rule:?} is marked as non-fixable but it created a fix.
|
||||
Change the `Violation::FIX_AVAILABILITY` to either \
|
||||
`FixAvailability::Sometimes` or `FixAvailability::Always`"
|
||||
);
|
||||
}
|
||||
(false, FixAvailability::Always) if source_has_errors => {
|
||||
// Ok
|
||||
}
|
||||
(false, FixAvailability::Always) => {
|
||||
panic!("Rule {rule:?} is marked to always-fixable but the diagnostic has no fix. Either ensure you always emit a fix or change `Violation::FIX_AVAILABILITY` to either `FixAvailability::Sometimes` or `FixAvailability::None")
|
||||
panic!(
|
||||
"\
|
||||
Rule {rule:?} is marked to always-fixable but the diagnostic has no fix.
|
||||
Either ensure you always emit a fix or change `Violation::FIX_AVAILABILITY` to either \
|
||||
`FixAvailability::Sometimes` or `FixAvailability::None`"
|
||||
)
|
||||
}
|
||||
}
|
||||
|
||||
assert!(!(fixable && diagnostic.kind.suggestion.is_none()), "Diagnostic emitted by {rule:?} is fixable but `Violation::fix_title` returns `None`.`");
|
||||
assert!(
|
||||
!(fixable && diagnostic.kind.suggestion.is_none()),
|
||||
"Diagnostic emitted by {rule:?} is fixable but \
|
||||
`Violation::fix_title` returns `None`"
|
||||
);
|
||||
|
||||
// Not strictly necessary but adds some coverage for this code path
|
||||
let noqa = directives.noqa_line_for.resolve(diagnostic.start());
|
||||
|
||||
Message::from_diagnostic(diagnostic, source_code.clone(), noqa)
|
||||
})
|
||||
.chain(
|
||||
parsed
|
||||
.errors()
|
||||
.iter()
|
||||
.map(|parse_error| {
|
||||
Message::from_parse_error(parse_error, &locator, source_code.clone())
|
||||
})
|
||||
)
|
||||
.chain(parsed.errors().iter().map(|parse_error| {
|
||||
Message::from_parse_error(parse_error, &locator, source_code.clone())
|
||||
}))
|
||||
.sorted()
|
||||
.collect();
|
||||
(messages, transformed)
|
||||
|
|
Loading…
Add table
Add a link
Reference in a new issue