From 2124feb0e71b55f7a3782944ff7dcf122ac21b2c Mon Sep 17 00:00:00 2001 From: Micha Reiser Date: Fri, 5 May 2023 07:59:33 +0200 Subject: [PATCH] Fail lint tests if the fix creates a syntax error (#4202) --- crates/ruff/src/test.rs | 112 +++++++++++++++++++++++----------------- 1 file changed, 66 insertions(+), 46 deletions(-) diff --git a/crates/ruff/src/test.rs b/crates/ruff/src/test.rs index f600f03c146f3..5fef8804b41ca 100644 --- a/crates/ruff/src/test.rs +++ b/crates/ruff/src/test.rs @@ -5,6 +5,7 @@ use std::path::Path; use anyhow::Result; use itertools::Itertools; +use ruff_diagnostics::Diagnostic; use rustc_hash::FxHashMap; use rustpython_parser::lexer::LexResult; @@ -15,6 +16,7 @@ use crate::directives; use crate::linter::{check_path, LinterResult}; use crate::message::{Emitter, EmitterContext, Message, TextEmitter}; use crate::packaging::detect_package_root; +use crate::rules::pycodestyle::rules::syntax_error; use crate::settings::{flags, Settings}; pub fn test_resource_path(path: impl AsRef) -> std::path::PathBuf { @@ -24,6 +26,8 @@ pub fn test_resource_path(path: impl AsRef) -> std::path::PathBuf { /// A convenient wrapper around [`check_path`], that additionally /// asserts that autofixes converge after 10 iterations. pub fn test_path(path: impl AsRef, settings: &Settings) -> Result> { + static MAX_ITERATIONS: usize = 10; + let path = test_resource_path("fixtures").join(path); let contents = std::fs::read_to_string(&path)?; let tokens: Vec = ruff_rustpython::tokenize(&contents); @@ -38,7 +42,7 @@ pub fn test_path(path: impl AsRef, settings: &Settings) -> Result, settings: &Settings) -> Result = ruff_rustpython::tokenize(&contents); - let locator = Locator::new(&contents); + while let Some((fixed_contents, _)) = fix_file(&diagnostics, &Locator::new(&contents)) { + if iterations < MAX_ITERATIONS { + iterations += 1; + } else { + let output = print_diagnostics(diagnostics, &path, &contents); + + panic!( + "Failed to converge after {MAX_ITERATIONS} iterations. This likely \ + indicates a bug in the implementation of the fix. Last diagnostics:\n{output}" + ); + } + + let tokens: Vec = ruff_rustpython::tokenize(&fixed_contents); + let locator = Locator::new(&fixed_contents); let stylist = Stylist::from_tokens(&tokens, &locator); let indexer = Indexer::from_tokens(&tokens, &locator); let directives = directives::extract_directives( @@ -74,9 +91,10 @@ pub fn test_path(path: impl AsRef, settings: &Settings) -> Result, settings: &Settings) -> Result = diagnostics - .into_iter() - .map(|diagnostic| { - // 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) - }) - .collect(); - - let mut output: Vec = Vec::new(); - TextEmitter::default() - .with_show_fix(true) - .with_show_source(true) - .emit( - &mut output, - &messages, - &EmitterContext::new(&FxHashMap::default()), - ) - .unwrap(); - - let output_str = String::from_utf8(output).unwrap(); + + if let Some(fixed_error) = fixed_error { + if !source_has_errors { + // Previous fix introduced a syntax error, abort + let fixes = print_diagnostics(diagnostics, &path, &contents); + + let mut syntax_diagnostics = Vec::new(); + syntax_error(&mut syntax_diagnostics, &fixed_error, &locator); + let syntax_errors = + print_diagnostics(syntax_diagnostics, &path, &fixed_contents); + panic!( - "Failed to converge after {max_iterations} iterations. This likely \ - indicates a bug in the implementation of the fix. Last diagnostics:\n{output_str}" + r#"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: +{fixed_contents}"# ); } - } else { - break; } + + diagnostics = fixed_diagnostics; + contents = fixed_contents.to_string(); } } @@ -151,6 +152,25 @@ pub fn test_path(path: impl AsRef, settings: &Settings) -> Result, file_path: &Path, source: &str) -> String { + let source_file = SourceFileBuilder::new( + file_path.file_name().unwrap().to_string_lossy().as_ref(), + source, + ) + .finish(); + + let messages: Vec<_> = diagnostics + .into_iter() + .map(|diagnostic| { + let noqa_start = diagnostic.start(); + + Message::from_diagnostic(diagnostic, source_file.clone(), noqa_start) + }) + .collect(); + + print_messages(&messages) +} + pub(crate) fn print_messages(messages: &[Message]) -> String { let mut output = Vec::new();