From 5d0fe4caff28748b2d58a693da86d5a31448c00a Mon Sep 17 00:00:00 2001 From: Ben Heidemann Date: Tue, 5 Nov 2024 16:39:19 +0000 Subject: [PATCH] fix: range coverage & add warning for unterminated ignore range directive --- .gitignore | 5 +- cli/tools/coverage/ignore_directives.rs | 42 ++++++++- cli/tools/coverage/mod.rs | 92 +++++++++++-------- tests/integration/coverage_tests.rs | 1 + .../ignore_next_directive_expected.lcov | 6 +- .../ignore_next_directive_expected.out | 2 +- .../ignore_range_directive_expected.lcov | 4 +- 7 files changed, 102 insertions(+), 50 deletions(-) diff --git a/.gitignore b/.gitignore index 8be7f979d1..9ae954d87f 100644 --- a/.gitignore +++ b/.gitignore @@ -30,9 +30,12 @@ gclient_config.py_entries /ext/websocket/autobahn/reports +# Coverage files +/coverage + # JUnit files produced by deno test --junit junit.xml # Jupyter files .ipynb_checkpoints/ -Untitled*.ipynb \ No newline at end of file +Untitled*.ipynb diff --git a/cli/tools/coverage/ignore_directives.rs b/cli/tools/coverage/ignore_directives.rs index b605f4d225..f090741833 100644 --- a/cli/tools/coverage/ignore_directives.rs +++ b/cli/tools/coverage/ignore_directives.rs @@ -8,6 +8,7 @@ use deno_ast::SourceRange; use deno_ast::SourceRanged; use deno_ast::SourceRangedForSpanned as _; use deno_ast::SourceTextInfoProvider as _; +use deno_core::url::Url; use std::collections::HashMap; static COVERAGE_IGNORE_START_DIRECTIVE: &str = "deno-coverage-ignore-start"; @@ -40,13 +41,21 @@ impl IgnoreDirective { } pub fn parse_range_ignore_directives( + is_quiet: bool, + script_module_specifier: &Url, program: &ast_view::Program, ) -> Vec { let mut depth: usize = 0; let mut directives = Vec::::new(); let mut current_range: Option = None; - for comment in program.comment_container().all_comments() { + let mut comments_sorted = program + .comment_container() + .all_comments() + .collect::>(); + comments_sorted.sort_by(|a, b| a.range().start.cmp(&b.range().start)); + + for comment in comments_sorted.iter() { if comment.kind != CommentKind::Line { continue; } @@ -77,6 +86,17 @@ pub fn parse_range_ignore_directives( // If the coverage ignore start directive has no corresponding close directive // then close it at the end of the program. if let Some(mut range) = current_range.take() { + if !is_quiet { + let text_info = program.text_info(); + let loc = text_info.line_and_column_display(range.start); + log::warn!( + "WARNING: Unterminated {} comment at {}:{}:{}", + COVERAGE_IGNORE_START_DIRECTIVE, + script_module_specifier, + loc.line_number, + loc.column_number, + ); + } range.end = program.range().end; directives.push(IgnoreDirective { range, @@ -164,6 +184,8 @@ fn parse_ignore_comment( #[cfg(test)] mod tests { + use std::str::FromStr; + use deno_ast::MediaType; use deno_ast::ModuleSpecifier; use deno_ast::ParsedSource; @@ -205,7 +227,11 @@ mod tests { "#; parse_and_then(source_code, |program| { - let line_directives = parse_range_ignore_directives(&program); + let line_directives = parse_range_ignore_directives( + true, + &Url::from_str("test.ts").unwrap(), + &program, + ); assert_eq!(line_directives.len(), 2); }); @@ -223,7 +249,11 @@ mod tests { "#; parse_and_then(source_code, |program| { - let line_directives = parse_range_ignore_directives(&program); + let line_directives = parse_range_ignore_directives( + true, + &Url::from_str("test.ts").unwrap(), + &program, + ); assert_eq!(line_directives.len(), 1); }); @@ -244,7 +274,11 @@ mod tests { "#; parse_and_then(source_code, |program| { - let line_directives = parse_range_ignore_directives(&program); + let line_directives = parse_range_ignore_directives( + true, + &Url::from_str("test.ts").unwrap(), + &program, + ); assert_eq!(line_directives.len(), 1); }); diff --git a/cli/tools/coverage/mod.rs b/cli/tools/coverage/mod.rs index e07d1a84d0..6b2d482e8a 100644 --- a/cli/tools/coverage/mod.rs +++ b/cli/tools/coverage/mod.rs @@ -201,24 +201,29 @@ pub struct CoverageReport { output: Option, } -fn generate_coverage_report( +struct GenerateCoverageReportOptions<'a> { + cli_options: &'a CliOptions, script_module_specifier: Url, script_media_type: MediaType, - script_coverage: &cdp::ScriptCoverage, + script_coverage: &'a cdp::ScriptCoverage, script_original_source: String, script_runtime_source: String, - maybe_source_map: &Option>, - output: &Option, + maybe_source_map: &'a Option>, + output: &'a Option, +} + +fn generate_coverage_report( + options: GenerateCoverageReportOptions, ) -> CoverageReport { let parsed_source = parse_program( - script_module_specifier, - script_media_type, - &script_original_source, + options.script_module_specifier, + options.script_media_type, + &options.script_original_source, ) .expect("invalid source code"); let ignore_file_directive = parsed_source.with_view(|program| parse_file_ignore_directives(&program)); - let url = Url::parse(&script_coverage.url).unwrap(); + let url = Url::parse(&options.script_coverage.url).unwrap(); if ignore_file_directive.is_some() { return CoverageReport { @@ -226,17 +231,18 @@ fn generate_coverage_report( named_functions: Vec::new(), branches: Vec::new(), found_lines: Vec::new(), - output: output.clone(), + output: options.output.clone(), }; } - let maybe_source_map = maybe_source_map + let maybe_source_map = options + .maybe_source_map .as_ref() .map(|source_map| SourceMap::from_slice(source_map).unwrap()); - let text_lines = TextLines::new(&script_runtime_source); + let text_lines = TextLines::new(&options.script_runtime_source); let comment_ranges = - deno_ast::lex(&script_runtime_source, MediaType::JavaScript) + deno_ast::lex(&options.script_runtime_source, MediaType::JavaScript) .into_iter() .filter(|item| { matches!(item.inner, deno_ast::TokenOrComment::Comment { .. }) @@ -247,7 +253,8 @@ fn generate_coverage_report( let mut coverage_report = CoverageReport { url, named_functions: Vec::with_capacity( - script_coverage + options + .script_coverage .functions .iter() .filter(|f| !f.function_name.is_empty()) @@ -255,24 +262,28 @@ fn generate_coverage_report( ), branches: Vec::new(), found_lines: Vec::new(), - output: output.clone(), + output: options.output.clone(), }; let coverage_ignore_next_directives = parsed_source.with_view(|program| parse_next_ignore_directives(&program)); let coverage_ignore_range_directives = parsed_source.with_view(|program| { - parse_range_ignore_directives(&program) - .iter() - .map(|directive| { - ( - program.text_info().line_index(directive.range().start), - program.text_info().line_index(directive.range().end), - ) - }) - .collect::>() + parse_range_ignore_directives( + options.cli_options.is_quiet(), + parsed_source.specifier(), + &program, + ) + .iter() + .map(|directive| { + ( + program.text_info().line_index(directive.range().start), + program.text_info().line_index(directive.range().end), + ) + }) + .collect::>() }); - for function in &script_coverage.functions { + for function in &options.script_coverage.functions { if function.function_name.is_empty() { continue; } @@ -303,7 +314,9 @@ fn generate_coverage_report( }); } - for (block_number, function) in script_coverage.functions.iter().enumerate() { + for (block_number, function) in + options.script_coverage.functions.iter().enumerate() + { let block_hits = function.ranges[0].count; for (branch_number, range) in function.ranges[1..].iter().enumerate() { let line_index = @@ -357,7 +370,7 @@ fn generate_coverage_report( let line_end_char_offset = text_lines.char_index(line_end_byte_offset); let ignore = comment_ranges.iter().any(|range| { range.start <= line_start_byte_offset && range.end >= line_end_byte_offset - }) || script_runtime_source + }) || options.script_runtime_source [line_start_byte_offset..line_end_byte_offset] .trim() .is_empty(); @@ -368,7 +381,7 @@ fn generate_coverage_report( } else { // Count the hits of ranges that include the entire line which will always be at-least one // as long as the code has been evaluated. - for function in &script_coverage.functions { + for function in &options.script_coverage.functions { for range in &function.ranges { if range.start_char_offset <= line_start_char_offset && range.end_char_offset >= line_end_char_offset @@ -379,7 +392,7 @@ fn generate_coverage_report( } // We reset the count if any block with a zero count overlaps with the line range. - for function in &script_coverage.functions { + for function in &options.script_coverage.functions { for range in &function.ranges { if range.count > 0 { continue; @@ -422,7 +435,8 @@ fn generate_coverage_report( coverage_report.found_lines = if let Some(source_map) = maybe_source_map.as_ref() { - let script_runtime_source_lines = script_runtime_source.lines().collect::>(); + let script_runtime_source_lines = + options.script_runtime_source.lines().collect::>(); let mut found_lines = line_counts .iter() .enumerate() @@ -718,15 +732,17 @@ pub fn cover_files( }; let source_map = source_map_from_code(runtime_code.as_bytes()); - let coverage_report = generate_coverage_report( - module_specifier, - file.media_type, - &script_coverage, - original_source.to_string(), - runtime_code.as_str().to_owned(), - &source_map, - &out_mode, - ); + let coverage_report = + generate_coverage_report(GenerateCoverageReportOptions { + cli_options, + script_module_specifier: module_specifier, + script_media_type: file.media_type, + script_coverage: &script_coverage, + script_original_source: original_source.to_string(), + script_runtime_source: runtime_code.as_str().to_owned(), + maybe_source_map: &source_map, + output: &out_mode, + }); if !coverage_report.found_lines.is_empty() { reporter.report(&coverage_report, &original_source)?; diff --git a/tests/integration/coverage_tests.rs b/tests/integration/coverage_tests.rs index cb3a4d79a4..5f90f315ae 100644 --- a/tests/integration/coverage_tests.rs +++ b/tests/integration/coverage_tests.rs @@ -134,6 +134,7 @@ fn run_coverage_text(test_name: &str, extension: &str) { .args_vec(vec![ "coverage".to_string(), "--detailed".to_string(), + "--quiet".to_string(), format!("{}/", tempdir), ]) .split_output() diff --git a/tests/testdata/coverage/ignore_next_directive_expected.lcov b/tests/testdata/coverage/ignore_next_directive_expected.lcov index d49840ff39..7818bbb9fb 100644 --- a/tests/testdata/coverage/ignore_next_directive_expected.lcov +++ b/tests/testdata/coverage/ignore_next_directive_expected.lcov @@ -5,11 +5,9 @@ FNF:1 FNH:1 BRF:0 BRH:0 -DA:1,1 -DA:2,1 DA:7,1 DA:11,2 DA:12,2 -LH:5 -LF:5 +LH:3 +LF:3 end_of_record diff --git a/tests/testdata/coverage/ignore_next_directive_expected.out b/tests/testdata/coverage/ignore_next_directive_expected.out index 54e4fef23a..df8ac11965 100644 --- a/tests/testdata/coverage/ignore_next_directive_expected.out +++ b/tests/testdata/coverage/ignore_next_directive_expected.out @@ -1 +1 @@ -cover [WILDCARD]ignore_next_directive.ts ... 100.000% (5/5) +cover [WILDCARD]ignore_next_directive.ts ... 100.000% (3/3) diff --git a/tests/testdata/coverage/ignore_range_directive_expected.lcov b/tests/testdata/coverage/ignore_range_directive_expected.lcov index 2bab40dc54..69cf27bad2 100644 --- a/tests/testdata/coverage/ignore_range_directive_expected.lcov +++ b/tests/testdata/coverage/ignore_range_directive_expected.lcov @@ -5,9 +5,9 @@ FNF:1 FNH:1 BRF:0 BRH:0 -DA:1,1 -DA:2,1 DA:7,1 +DA:14,2 +DA:15,2 LH:3 LF:3 end_of_record