Skip to content

Commit 020db0d

Browse files
committed
Don't require all build script output to be utf-8
Build scripts often stream output of native build systems like cmake/make and those aren't always guaranteed to produce utf-8 output. For example German MSVC cmake build has been reported to print non-utf-8 umlauts. This commit instead only attempts to interpret each line as utf-8 rather than the entire build script output. All non-utf-8 output is ignored. Closes #2556
1 parent 10ddd7d commit 020db0d

File tree

3 files changed

+54
-9
lines changed

3 files changed

+54
-9
lines changed

src/cargo/ops/cargo_rustc/custom_build.rs

+9-8
Original file line numberDiff line numberDiff line change
@@ -6,7 +6,7 @@ use std::str;
66
use std::sync::{Mutex, Arc};
77

88
use core::PackageId;
9-
use util::{CargoResult, human, Human};
9+
use util::{CargoResult, Human};
1010
use util::{internal, ChainError, profile, paths};
1111
use util::Freshness;
1212

@@ -213,10 +213,7 @@ fn build_work<'a, 'cfg>(cx: &mut Context<'a, 'cfg>, unit: &Unit<'a>)
213213
// This is also the location where we provide feedback into the build
214214
// state informing what variables were discovered via our script as
215215
// well.
216-
let output = try!(str::from_utf8(&output.stdout).map_err(|_| {
217-
human("build script output was not valid utf-8")
218-
}));
219-
let parsed_output = try!(BuildOutput::parse(output, &pkg_name));
216+
let parsed_output = try!(BuildOutput::parse(&output.stdout, &pkg_name));
220217
build_state.insert(id, kind, parsed_output);
221218
Ok(())
222219
});
@@ -270,21 +267,25 @@ impl BuildState {
270267

271268
impl BuildOutput {
272269
pub fn parse_file(path: &Path, pkg_name: &str) -> CargoResult<BuildOutput> {
273-
let contents = try!(paths::read(path));
270+
let contents = try!(paths::read_bytes(path));
274271
BuildOutput::parse(&contents, pkg_name)
275272
}
276273

277274
// Parses the output of a script.
278275
// The `pkg_name` is used for error messages.
279-
pub fn parse(input: &str, pkg_name: &str) -> CargoResult<BuildOutput> {
276+
pub fn parse(input: &[u8], pkg_name: &str) -> CargoResult<BuildOutput> {
280277
let mut library_paths = Vec::new();
281278
let mut library_links = Vec::new();
282279
let mut cfgs = Vec::new();
283280
let mut metadata = Vec::new();
284281
let mut rerun_if_changed = Vec::new();
285282
let whence = format!("build script of `{}`", pkg_name);
286283

287-
for line in input.lines() {
284+
for line in input.split(|b| *b == b'\n') {
285+
let line = match str::from_utf8(line) {
286+
Ok(line) => line.trim(),
287+
Err(..) => continue,
288+
};
288289
let mut iter = line.splitn(2, ':');
289290
if iter.next() != Some("cargo") {
290291
// skip this line since it doesn't start with "cargo:"

src/cargo/util/paths.rs

+12-1
Original file line numberDiff line numberDiff line change
@@ -68,7 +68,7 @@ pub fn without_prefix<'a>(a: &'a Path, b: &'a Path) -> Option<&'a Path> {
6868
}
6969

7070
pub fn read(path: &Path) -> CargoResult<String> {
71-
(|| -> CargoResult<String> {
71+
(|| -> CargoResult<_> {
7272
let mut ret = String::new();
7373
let mut f = try!(File::open(path));
7474
try!(f.read_to_string(&mut ret));
@@ -78,6 +78,17 @@ pub fn read(path: &Path) -> CargoResult<String> {
7878
})
7979
}
8080

81+
pub fn read_bytes(path: &Path) -> CargoResult<Vec<u8>> {
82+
(|| -> CargoResult<_> {
83+
let mut ret = Vec::new();
84+
let mut f = try!(File::open(path));
85+
try!(f.read_to_end(&mut ret));
86+
Ok(ret)
87+
})().map_err(human).chain_error(|| {
88+
human(format!("failed to read `{}`", path.display()))
89+
})
90+
}
91+
8192
pub fn write(path: &Path, contents: &[u8]) -> CargoResult<()> {
8293
(|| -> CargoResult<()> {
8394
let mut f = try!(File::create(path));

tests/test_cargo_compile_custom_build.rs

+33
Original file line numberDiff line numberDiff line change
@@ -1866,3 +1866,36 @@ test!(please_respect_the_dag {
18661866
{running} `rustc [..] -L native=foo -L native=bar[..]`
18671867
", running = RUNNING)));
18681868
});
1869+
1870+
test!(non_utf8_output {
1871+
let p = project("foo")
1872+
.file("Cargo.toml", r#"
1873+
[project]
1874+
name = "foo"
1875+
version = "0.5.0"
1876+
authors = []
1877+
build = "build.rs"
1878+
"#)
1879+
.file("build.rs", r#"
1880+
use std::io::prelude::*;
1881+
1882+
fn main() {
1883+
let mut out = std::io::stdout();
1884+
// print something that's not utf8
1885+
out.write_all(b"\xff\xff\n").unwrap();
1886+
1887+
// now print some cargo metadata that's utf8
1888+
println!("cargo:rustc-cfg=foo");
1889+
1890+
// now print more non-utf8
1891+
out.write_all(b"\xff\xff\n").unwrap();
1892+
}
1893+
"#)
1894+
.file("src/main.rs", r#"
1895+
#[cfg(foo)]
1896+
fn main() {}
1897+
"#);
1898+
1899+
assert_that(p.cargo_process("build").arg("-v"),
1900+
execs().with_status(0));
1901+
});

0 commit comments

Comments
 (0)