Skip to content

Commit 4d63435

Browse files
committed
Auto merge of #76196 - r-52:r-coverage-allow-missing-docs, r=jyn514
rustdoc: skip #[allow(missing docs)] for docs in coverage report During the document coverage reporting with: ```bash rustdoc something.rs -Z unstable-options --show-coverage ``` the coverage report counts code that is marked with `#[allow(missing_docs)]` for the calculation, which outputs lower numbers in the coverage report even though these parts should be ignored for the calculation. Right now I'm not sure how this can be tested (CI)? (I verified it by hand and ran the unit tests) r? `@jyn514` **Reference:** Fixes #76121
2 parents f54072b + 6854440 commit 4d63435

File tree

5 files changed

+104
-15
lines changed

5 files changed

+104
-15
lines changed

src/librustdoc/passes/calculate_doc_coverage.rs

+25-11
Original file line numberDiff line numberDiff line change
@@ -1,10 +1,12 @@
11
use crate::clean;
2-
use crate::config::OutputFormat;
32
use crate::core::DocContext;
43
use crate::fold::{self, DocFolder};
54
use crate::html::markdown::{find_testable_code, ErrorCodes};
65
use crate::passes::doc_test_lints::{should_have_doc_example, Tests};
76
use crate::passes::Pass;
7+
use rustc_lint::builtin::MISSING_DOCS;
8+
use rustc_middle::lint::LintSource;
9+
use rustc_session::lint;
810
use rustc_span::symbol::sym;
911
use rustc_span::FileName;
1012
use serde::Serialize;
@@ -19,10 +21,10 @@ pub const CALCULATE_DOC_COVERAGE: Pass = Pass {
1921
};
2022

2123
fn calculate_doc_coverage(krate: clean::Crate, ctx: &DocContext<'_>) -> clean::Crate {
22-
let mut calc = CoverageCalculator::new();
24+
let mut calc = CoverageCalculator::new(ctx);
2325
let krate = calc.fold_crate(krate);
2426

25-
calc.print_results(ctx.renderinfo.borrow().output_format);
27+
calc.print_results();
2628

2729
krate
2830
}
@@ -41,8 +43,11 @@ impl ItemCount {
4143
has_docs: bool,
4244
has_doc_example: bool,
4345
should_have_doc_examples: bool,
46+
should_have_docs: bool,
4447
) {
45-
self.total += 1;
48+
if has_docs || should_have_docs {
49+
self.total += 1;
50+
}
4651

4752
if has_docs {
4853
self.with_docs += 1;
@@ -94,8 +99,9 @@ impl ops::AddAssign for ItemCount {
9499
}
95100
}
96101

97-
struct CoverageCalculator {
102+
struct CoverageCalculator<'a, 'b> {
98103
items: BTreeMap<FileName, ItemCount>,
104+
ctx: &'a DocContext<'b>,
99105
}
100106

101107
fn limit_filename_len(filename: String) -> String {
@@ -108,9 +114,9 @@ fn limit_filename_len(filename: String) -> String {
108114
}
109115
}
110116

111-
impl CoverageCalculator {
112-
fn new() -> CoverageCalculator {
113-
CoverageCalculator { items: Default::default() }
117+
impl<'a, 'b> CoverageCalculator<'a, 'b> {
118+
fn new(ctx: &'a DocContext<'b>) -> CoverageCalculator<'a, 'b> {
119+
CoverageCalculator { items: Default::default(), ctx }
114120
}
115121

116122
fn to_json(&self) -> String {
@@ -124,7 +130,8 @@ impl CoverageCalculator {
124130
.expect("failed to convert JSON data to string")
125131
}
126132

127-
fn print_results(&self, output_format: Option<OutputFormat>) {
133+
fn print_results(&self) {
134+
let output_format = self.ctx.renderinfo.borrow().output_format;
128135
if output_format.map(|o| o.is_json()).unwrap_or_else(|| false) {
129136
println!("{}", self.to_json());
130137
return;
@@ -178,7 +185,7 @@ impl CoverageCalculator {
178185
}
179186
}
180187

181-
impl fold::DocFolder for CoverageCalculator {
188+
impl<'a, 'b> fold::DocFolder for CoverageCalculator<'a, 'b> {
182189
fn fold_item(&mut self, i: clean::Item) -> Option<clean::Item> {
183190
match i.inner {
184191
_ if !i.def_id.is_local() => {
@@ -245,11 +252,18 @@ impl fold::DocFolder for CoverageCalculator {
245252
);
246253

247254
let has_doc_example = tests.found_tests != 0;
255+
let hir_id = self.ctx.tcx.hir().local_def_id_to_hir_id(i.def_id.expect_local());
256+
let (level, source) = self.ctx.tcx.lint_level_at_node(MISSING_DOCS, hir_id);
257+
// `missing_docs` is allow-by-default, so don't treat this as ignoring the item
258+
// unless the user had an explicit `allow`
259+
let should_have_docs =
260+
level != lint::Level::Allow || matches!(source, LintSource::Default);
248261
debug!("counting {:?} {:?} in {}", i.type_(), i.name, i.source.filename);
249262
self.items.entry(i.source.filename.clone()).or_default().count_item(
250263
has_docs,
251264
has_doc_example,
252-
should_have_doc_example(&i.inner),
265+
should_have_doc_example(self.ctx, &i),
266+
should_have_docs,
253267
);
254268
}
255269
}

src/librustdoc/passes/doc_test_lints.rs

+11-4
Original file line numberDiff line numberDiff line change
@@ -9,6 +9,7 @@ use crate::clean::*;
99
use crate::core::DocContext;
1010
use crate::fold::DocFolder;
1111
use crate::html::markdown::{find_testable_code, ErrorCodes, Ignore, LangString};
12+
use rustc_middle::lint::LintSource;
1213
use rustc_session::lint;
1314

1415
pub const CHECK_PRIVATE_ITEMS_DOC_TESTS: Pass = Pass {
@@ -56,8 +57,8 @@ impl crate::doctest::Tester for Tests {
5657
}
5758
}
5859

59-
pub fn should_have_doc_example(item_kind: &clean::ItemEnum) -> bool {
60-
!matches!(item_kind,
60+
pub fn should_have_doc_example(cx: &DocContext<'_>, item: &clean::Item) -> bool {
61+
if matches!(item.inner,
6162
clean::StructFieldItem(_)
6263
| clean::VariantItem(_)
6364
| clean::AssocConstItem(_, _)
@@ -69,7 +70,13 @@ pub fn should_have_doc_example(item_kind: &clean::ItemEnum) -> bool {
6970
| clean::ImportItem(_)
7071
| clean::PrimitiveItem(_)
7172
| clean::KeywordItem(_)
72-
)
73+
) {
74+
return false;
75+
}
76+
let hir_id = cx.tcx.hir().local_def_id_to_hir_id(item.def_id.expect_local());
77+
let (level, source) =
78+
cx.tcx.lint_level_at_node(lint::builtin::MISSING_DOC_CODE_EXAMPLES, hir_id);
79+
level != lint::Level::Allow || matches!(source, LintSource::Default)
7380
}
7481

7582
pub fn look_for_tests<'tcx>(cx: &DocContext<'tcx>, dox: &str, item: &Item) {
@@ -88,7 +95,7 @@ pub fn look_for_tests<'tcx>(cx: &DocContext<'tcx>, dox: &str, item: &Item) {
8895
if tests.found_tests == 0
8996
&& rustc_feature::UnstableFeatures::from_environment().is_nightly_build()
9097
{
91-
if should_have_doc_example(&item.inner) {
98+
if should_have_doc_example(cx, &item) {
9299
debug!("reporting error for {:?} (hir_id={:?})", item, hir_id);
93100
let sp = span_of_attrs(&item.attrs).unwrap_or(item.source.span());
94101
cx.tcx.struct_span_lint_hir(
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,41 @@
1+
// compile-flags:-Z unstable-options --show-coverage
2+
// check-pass
3+
4+
//! Make sure to have some docs on your crate root
5+
6+
#[allow(missing_docs)]
7+
pub mod mod_foo {
8+
pub struct Bar;
9+
}
10+
11+
/// This is a struct with a `#[allow(missing_docs)]`
12+
pub struct AllowTheMissingDocs {
13+
#[allow(missing_docs)]
14+
pub empty_str: String,
15+
16+
/// This has
17+
#[allow(missing_docs)]
18+
/// but also has documentation comments
19+
pub hello: usize,
20+
21+
/// The doc id just to create a boilerplate comment
22+
pub doc_id: Vec<u8>,
23+
}
24+
25+
/// A function that has a documentation
26+
pub fn this_is_func() {}
27+
28+
#[allow(missing_docs)]
29+
pub struct DemoStruct {
30+
something: usize,
31+
}
32+
33+
#[allow(missing_docs)]
34+
pub mod bar {
35+
#[warn(missing_docs)]
36+
pub struct Bar { //~ WARN
37+
pub f: u32, //~ WARN
38+
}
39+
40+
pub struct NeedsNoDocs;
41+
}
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,20 @@
1+
warning: missing documentation for a struct
2+
--> $DIR/allow_missing_docs.rs:36:5
3+
|
4+
LL | pub struct Bar {
5+
| ^^^^^^^^^^^^^^
6+
|
7+
note: the lint level is defined here
8+
--> $DIR/allow_missing_docs.rs:35:12
9+
|
10+
LL | #[warn(missing_docs)]
11+
| ^^^^^^^^^^^^
12+
13+
warning: missing documentation for a struct field
14+
--> $DIR/allow_missing_docs.rs:37:9
15+
|
16+
LL | pub f: u32,
17+
| ^^^^^^^^^^
18+
19+
warning: 2 warnings emitted
20+
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,7 @@
1+
+-------------------------------------+------------+------------+------------+------------+
2+
| File | Documented | Percentage | Examples | Percentage |
3+
+-------------------------------------+------------+------------+------------+------------+
4+
| ...i/coverage/allow_missing_docs.rs | 5 | 71.4% | 0 | 0.0% |
5+
+-------------------------------------+------------+------------+------------+------------+
6+
| Total | 5 | 71.4% | 0 | 0.0% |
7+
+-------------------------------------+------------+------------+------------+------------+

0 commit comments

Comments
 (0)