Skip to content

Commit 109d9c5

Browse files
committed
Auto merge of #3410 - fmdkdd:fix-3390-message-format-parse-stderr, r=alexcrichton
Fix `--message-format JSON` when rustc emits non-JSON warnings The `--message-format JSON` flag parses all the stderr output of rustc to JSON, but rustc can emit non-JSON lines to stderr (e.g., for warning about the unstable `-Z` flag on the stable channel), causing cargo to fail reporting compilation errors when using `--message-format JSON`. This commit adds a check to look for lines beginning with `{` to only parse these lines as JSON. Other lines from rustc are forwarded to the stderr of cargo. Fixes #3390.
2 parents ddb5c32 + 3e32503 commit 109d9c5

File tree

2 files changed

+66
-9
lines changed

2 files changed

+66
-9
lines changed

src/cargo/ops/cargo_rustc/mod.rs

+17-9
Original file line numberDiff line numberDiff line change
@@ -2,6 +2,7 @@ use std::collections::{HashMap, HashSet};
22
use std::env;
33
use std::ffi::{OsStr, OsString};
44
use std::fs;
5+
use std::io::{self, Write};
56
use std::path::{self, PathBuf};
67
use std::sync::Arc;
78

@@ -297,15 +298,22 @@ fn rustc(cx: &mut Context, unit: &Unit) -> CargoResult<Work> {
297298
Ok(())
298299
},
299300
&mut |line| {
300-
let compiler_message = json::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-
});
301+
// stderr from rustc can have a mix of JSON and non-JSON output
302+
if line.starts_with("{") {
303+
// Handle JSON lines
304+
let compiler_message = json::Json::from_str(line).map_err(|_| {
305+
internal(&format!("compiler produced invalid json: `{}`", line))
306+
})?;
307+
308+
machine_message::emit(machine_message::FromCompiler {
309+
package_id: &package_id,
310+
target: &target,
311+
message: compiler_message,
312+
});
313+
} else {
314+
// Forward non-JSON to stderr
315+
writeln!(io::stderr(), "{}", line)?;
316+
}
309317
Ok(())
310318
},
311319
).map(|_| ())

tests/build.rs

+49
Original file line numberDiff line numberDiff line change
@@ -2470,6 +2470,55 @@ fn wrong_message_format_option() {
24702470
r#"[ERROR] Could not match 'xml' with any of the allowed variants: ["Human", "Json"]"#));
24712471
}
24722472

2473+
#[test]
2474+
fn message_format_json_forward_stderr() {
2475+
if is_nightly() { return }
2476+
2477+
let p = project("foo")
2478+
.file("Cargo.toml", &basic_bin_manifest("foo"))
2479+
.file("src/main.rs", "fn main() { let unused = 0; }");
2480+
2481+
assert_that(p.cargo_process("rustc").arg("--bin").arg("foo")
2482+
.arg("--message-format").arg("JSON").arg("--").arg("-Zno-trans"),
2483+
execs()
2484+
.with_stderr_contains("[WARNING] the option `Z` is unstable [..]")
2485+
.with_json(r#"
2486+
{
2487+
"reason":"compiler-message",
2488+
"package_id":"foo 0.5.0 ([..])",
2489+
"target":{"kind":["bin"],"name":"foo","src_path":"[..]"},
2490+
"message":{
2491+
"children":[],"code":null,"level":"warning","rendered":null,
2492+
"message":"unused variable: `unused`, #[warn(unused_variables)] on by default",
2493+
"spans":[{
2494+
"byte_end":22,"byte_start":16,"column_end":23,"column_start":17,"expansion":null,
2495+
"file_name":"[..]","is_primary":true,"label":null,"line_end":1,"line_start":1,
2496+
"suggested_replacement":null,
2497+
"text":[{
2498+
"highlight_end":23,
2499+
"highlight_start":17,
2500+
"text":"fn main() { let unused = 0; }"
2501+
}]
2502+
}]
2503+
}
2504+
}
2505+
2506+
{
2507+
"reason":"compiler-artifact",
2508+
"package_id":"foo 0.5.0 ([..])",
2509+
"target":{"kind":["bin"],"name":"foo","src_path":"[..]"},
2510+
"profile":{
2511+
"debug_assertions":true,
2512+
"debuginfo":true,
2513+
"opt_level":"0",
2514+
"test":false
2515+
},
2516+
"features":[],
2517+
"filenames":["[..]"]
2518+
}
2519+
"#));
2520+
}
2521+
24732522
#[test]
24742523
fn no_warn_about_package_metadata() {
24752524
let p = project("foo")

0 commit comments

Comments
 (0)