Skip to content

Commit 4adae9e

Browse files
committed
Always emit build script warnings for crates that fail to build
Resolves #3777
1 parent 0f697fa commit 4adae9e

File tree

2 files changed

+120
-6
lines changed

2 files changed

+120
-6
lines changed

src/cargo/ops/cargo_rustc/job_queue.rs

+28-6
Original file line numberDiff line numberDiff line change
@@ -182,13 +182,17 @@ impl<'a> JobQueue<'a> {
182182
match result {
183183
Ok(()) => self.finish(key, cx)?,
184184
Err(e) => {
185+
let msg = "The following warnings were emitted during compilation:";
186+
self.emit_warnings(Some(msg), key, cx)?;
187+
185188
if self.active > 0 {
186189
error = Some(human("build failed"));
187190
handle_error(&*e, &mut *cx.config.shell());
188191
cx.config.shell().say(
189192
"Build failed, waiting for other \
190193
jobs to finish...", YELLOW)?;
191194
}
195+
192196
if error.is_none() {
193197
error = Some(e);
194198
}
@@ -252,15 +256,33 @@ impl<'a> JobQueue<'a> {
252256
Ok(())
253257
}
254258

255-
fn finish(&mut self, key: Key<'a>, cx: &mut Context) -> CargoResult<()> {
256-
if key.profile.run_custom_build && cx.show_warnings(key.pkg) {
257-
let output = cx.build_state.outputs.lock().unwrap();
258-
if let Some(output) = output.get(&(key.pkg.clone(), key.kind)) {
259-
for warning in output.warnings.iter() {
260-
cx.config.shell().warn(warning)?;
259+
fn emit_warnings(&self, msg: Option<&str>, key: Key<'a>, cx: &mut Context) -> CargoResult<()> {
260+
let output = cx.build_state.outputs.lock().unwrap();
261+
if let Some(output) = output.get(&(key.pkg.clone(), key.kind)) {
262+
if let Some(msg) = msg {
263+
if !output.warnings.is_empty() {
264+
writeln!(cx.config.shell().err(), "{}\n", msg)?;
261265
}
262266
}
267+
268+
for warning in output.warnings.iter() {
269+
cx.config.shell().warn(warning)?;
270+
}
271+
272+
if !output.warnings.is_empty() && msg.is_some() {
273+
// Output an empty line.
274+
writeln!(cx.config.shell().err(), "")?;
275+
}
263276
}
277+
278+
Ok(())
279+
}
280+
281+
fn finish(&mut self, key: Key<'a>, cx: &mut Context) -> CargoResult<()> {
282+
if key.profile.run_custom_build && cx.show_warnings(key.pkg) {
283+
self.emit_warnings(None, key, cx)?;
284+
}
285+
264286
let state = self.pending.get_mut(&key).unwrap();
265287
state.amt -= 1;
266288
if state.amt == 0 {

tests/warn-on-failure.rs

+92
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,92 @@
1+
extern crate cargotest;
2+
extern crate hamcrest;
3+
4+
use cargotest::support::{project, execs, ProjectBuilder};
5+
use cargotest::support::registry::Package;
6+
use hamcrest::assert_that;
7+
8+
static WARNING1: &'static str = "Hello! I'm a warning. :)";
9+
static WARNING2: &'static str = "And one more!";
10+
11+
fn make_lib(lib_src: &str) {
12+
Package::new("foo", "0.0.1")
13+
.file("Cargo.toml", r#"
14+
[package]
15+
name = "foo"
16+
authors = []
17+
version = "0.0.1"
18+
build = "build.rs"
19+
"#)
20+
.file("build.rs", &format!(r#"
21+
fn main() {{
22+
use std::io::Write;
23+
println!("cargo:warning={{}}", "{}");
24+
println!("hidden stdout");
25+
write!(&mut ::std::io::stderr(), "hidden stderr");
26+
println!("cargo:warning={{}}", "{}");
27+
}}
28+
"#, WARNING1, WARNING2))
29+
.file("src/lib.rs", &format!("fn f() {{ {} }}", lib_src))
30+
.publish();
31+
}
32+
33+
fn make_upstream(main_src: &str) -> ProjectBuilder {
34+
project("bar")
35+
.file("Cargo.toml", r#"
36+
[package]
37+
name = "bar"
38+
version = "0.0.1"
39+
authors = []
40+
41+
[dependencies]
42+
foo = "*"
43+
"#)
44+
.file("src/main.rs", &format!("fn main() {{ {} }}", main_src))
45+
}
46+
47+
#[test]
48+
fn no_warning_on_success() {
49+
make_lib("");
50+
let upstream = make_upstream("");
51+
assert_that(upstream.cargo_process("build"),
52+
execs().with_status(0)
53+
.with_stderr("\
54+
[UPDATING] registry `[..]`
55+
[DOWNLOADING] foo v0.0.1 ([..])
56+
[COMPILING] foo v0.0.1
57+
[COMPILING] bar v0.0.1 ([..])
58+
[FINISHED] dev [unoptimized + debuginfo] target(s) in [..]
59+
"));
60+
}
61+
62+
#[test]
63+
fn no_warning_on_bin_failure() {
64+
make_lib("");
65+
let upstream = make_upstream("hi()");
66+
assert_that(upstream.cargo_process("build"),
67+
execs().with_status(101)
68+
.with_stdout_does_not_contain("hidden stdout")
69+
.with_stderr_does_not_contain("hidden stderr")
70+
.with_stderr_does_not_contain(&format!("[WARNING] {}", WARNING1))
71+
.with_stderr_does_not_contain(&format!("[WARNING] {}", WARNING2))
72+
.with_stderr_contains("[UPDATING] registry `[..]`")
73+
.with_stderr_contains("[DOWNLOADING] foo v0.0.1 ([..])")
74+
.with_stderr_contains("[COMPILING] foo v0.0.1")
75+
.with_stderr_contains("[COMPILING] bar v0.0.1 ([..])"));
76+
}
77+
78+
#[test]
79+
fn warning_on_lib_failure() {
80+
make_lib("err()");
81+
let upstream = make_upstream("");
82+
assert_that(upstream.cargo_process("build"),
83+
execs().with_status(101)
84+
.with_stdout_does_not_contain("hidden stdout")
85+
.with_stderr_does_not_contain("hidden stderr")
86+
.with_stderr_does_not_contain("[COMPILING] bar v0.0.1 ([..])")
87+
.with_stderr_contains("[UPDATING] registry `[..]`")
88+
.with_stderr_contains("[DOWNLOADING] foo v0.0.1 ([..])")
89+
.with_stderr_contains("[COMPILING] foo v0.0.1")
90+
.with_stderr_contains(&format!("[WARNING] {}", WARNING1))
91+
.with_stderr_contains(&format!("[WARNING] {}", WARNING2)));
92+
}

0 commit comments

Comments
 (0)