Skip to content

Commit 578e253

Browse files
committed
Auto merge of #5878 - ehuss:rustdoc-json, r=alexcrichton
Support JSON with rustdoc. This allows `cargo doc --message-format=json` to actually work. Note that this explicitly does not attempt to support it for doctests for several reasons: - `rustdoc --test --error-format=json` does not work for some reason. - Since the lib is usually compiled before running rustdoc, warnings/errors will be emitted correctly by rustc. - I'm unaware of any errors/warnings `rustdoc --test` is capable of producing assuming the code passed `rustc`. - The compilation of the tests themselves do not support JSON. - libtest does not output json, so it's utility is limited.
2 parents 1778152 + c28823f commit 578e253

File tree

2 files changed

+87
-40
lines changed

2 files changed

+87
-40
lines changed

src/cargo/core/compiler/mod.rs

+49-31
Original file line numberDiff line numberDiff line change
@@ -283,36 +283,10 @@ fn rustc<'a, 'cfg>(
283283
&package_id,
284284
&target,
285285
mode,
286-
&mut |line| {
287-
if !line.is_empty() {
288-
Err(internal(&format!(
289-
"compiler stdout is not empty: `{}`",
290-
line
291-
)))
292-
} else {
293-
Ok(())
294-
}
295-
},
296-
&mut |line| {
297-
// stderr from rustc can have a mix of JSON and non-JSON output
298-
if line.starts_with('{') {
299-
// Handle JSON lines
300-
let compiler_message = serde_json::from_str(line).map_err(|_| {
301-
internal(&format!("compiler produced invalid json: `{}`", line))
302-
})?;
303-
304-
machine_message::emit(&machine_message::FromCompiler {
305-
package_id: &package_id,
306-
target: &target,
307-
message: compiler_message,
308-
});
309-
} else {
310-
// Forward non-JSON to stderr
311-
writeln!(io::stderr(), "{}", line)?;
312-
}
313-
Ok(())
314-
},
315-
).map_err(Internal::new).chain_err(|| format!("Could not compile `{}`.", name))?;
286+
&mut assert_is_empty,
287+
&mut |line| json_stderr(line, &package_id, &target),
288+
).map_err(Internal::new)
289+
.chain_err(|| format!("Could not compile `{}`.", name))?;
316290
} else if build_plan {
317291
state.build_plan(buildkey, rustc.clone(), outputs.clone());
318292
} else {
@@ -631,6 +605,10 @@ fn rustdoc<'a, 'cfg>(cx: &mut Context<'a, 'cfg>, unit: &Unit<'a>) -> CargoResult
631605
rustdoc.arg("--cfg").arg(&format!("feature=\"{}\"", feat));
632606
}
633607

608+
if bcx.build_config.json_messages() {
609+
rustdoc.arg("--error-format").arg("json");
610+
}
611+
634612
if let Some(ref args) = bcx.extra_args_for(unit) {
635613
rustdoc.args(args);
636614
}
@@ -642,6 +620,9 @@ fn rustdoc<'a, 'cfg>(cx: &mut Context<'a, 'cfg>, unit: &Unit<'a>) -> CargoResult
642620
let name = unit.pkg.name().to_string();
643621
let build_state = cx.build_state.clone();
644622
let key = (unit.pkg.package_id().clone(), unit.kind);
623+
let json_messages = bcx.build_config.json_messages();
624+
let package_id = unit.pkg.package_id().clone();
625+
let target = unit.target.clone();
645626

646627
let should_capture_output = cx.bcx.config.cli_unstable().compile_progress;
647628

@@ -656,7 +637,14 @@ fn rustdoc<'a, 'cfg>(cx: &mut Context<'a, 'cfg>, unit: &Unit<'a>) -> CargoResult
656637
}
657638
state.running(&rustdoc);
658639

659-
let exec_result = if should_capture_output {
640+
let exec_result = if json_messages {
641+
rustdoc
642+
.exec_with_streaming(
643+
&mut assert_is_empty,
644+
&mut |line| json_stderr(line, &package_id, &target),
645+
false,
646+
).map(drop)
647+
} else if should_capture_output {
660648
state.capture_output(rustdoc, false).map(drop)
661649
} else {
662650
rustdoc.exec()
@@ -999,3 +987,33 @@ impl Kind {
999987
}
1000988
}
1001989
}
990+
991+
fn assert_is_empty(line: &str) -> CargoResult<()> {
992+
if !line.is_empty() {
993+
Err(internal(&format!(
994+
"compiler stdout is not empty: `{}`",
995+
line
996+
)))
997+
} else {
998+
Ok(())
999+
}
1000+
}
1001+
1002+
fn json_stderr(line: &str, package_id: &PackageId, target: &Target) -> CargoResult<()> {
1003+
// stderr from rustc/rustdoc can have a mix of JSON and non-JSON output
1004+
if line.starts_with('{') {
1005+
// Handle JSON lines
1006+
let compiler_message = serde_json::from_str(line)
1007+
.map_err(|_| internal(&format!("compiler produced invalid json: `{}`", line)))?;
1008+
1009+
machine_message::emit(&machine_message::FromCompiler {
1010+
package_id: package_id,
1011+
target: target,
1012+
message: compiler_message,
1013+
});
1014+
} else {
1015+
// Forward non-JSON to stderr
1016+
writeln!(io::stderr(), "{}", line)?;
1017+
}
1018+
Ok(())
1019+
}

tests/testsuite/doc.rs

+38-9
Original file line numberDiff line numberDiff line change
@@ -1359,22 +1359,22 @@ fn doc_private_items() {
13591359
assert_that(&foo.root().join("target/doc/foo/private/index.html"), existing_file());
13601360
}
13611361

1362+
const BAD_INTRA_LINK_LIB: &'static str = r#"
1363+
#![deny(intra_doc_link_resolution_failure)]
1364+
1365+
/// [bad_link]
1366+
pub fn foo() {}
1367+
"#;
1368+
13621369
#[test]
13631370
fn doc_cap_lints() {
13641371
if !is_nightly() {
13651372
// This can be removed once 1.29 is stable (rustdoc --cap-lints).
13661373
return;
13671374
}
13681375
let a = git::new("a", |p| {
1369-
p.file("Cargo.toml", &basic_lib_manifest("a")).file(
1370-
"src/lib.rs",
1371-
"
1372-
#![deny(intra_doc_link_resolution_failure)]
1373-
1374-
/// [bad_link]
1375-
pub fn foo() {}
1376-
",
1377-
)
1376+
p.file("Cargo.toml", &basic_lib_manifest("a"))
1377+
.file("src/lib.rs", BAD_INTRA_LINK_LIB)
13781378
}).unwrap();
13791379

13801380
let p = project()
@@ -1419,5 +1419,34 @@ fn doc_cap_lints() {
14191419
",
14201420
),
14211421
);
1422+
}
14221423

1424+
#[test]
1425+
fn doc_message_format() {
1426+
if !is_nightly() {
1427+
// This can be removed once 1.30 is stable (rustdoc --error-format stabilized).
1428+
return;
1429+
}
1430+
let p = project().file("src/lib.rs", BAD_INTRA_LINK_LIB).build();
1431+
1432+
assert_that(
1433+
p.cargo("doc --message-format=json"),
1434+
execs().with_status(101).with_json(
1435+
r#"
1436+
{
1437+
"message": {
1438+
"children": "{...}",
1439+
"code": "{...}",
1440+
"level": "error",
1441+
"message": "[..]",
1442+
"rendered": "[..]",
1443+
"spans": "{...}"
1444+
},
1445+
"package_id": "foo [..]",
1446+
"reason": "compiler-message",
1447+
"target": "{...}"
1448+
}
1449+
"#,
1450+
),
1451+
);
14231452
}

0 commit comments

Comments
 (0)