Skip to content

Commit 333a798

Browse files
committed
Auto merge of #3361 - pwoolcoc:default-build-script, r=alexcrichton
Assume `build.rs` in the same directory as `Cargo.toml` is a build script (unless explicitly told not to) So, in May I posted a question in the #cargo IRC room: https://botbot.me/mozilla/cargo/2016-05-26/?msg=66770068&page=1 This PR does what was discussed there. If `cargo` sees a `build.rs` in the same directory as the `Cargo.toml`, it will assume `build.rs` is a build script unless `package.build = false`. Just for completeness I also made `build = true` mean the same as `build = "build.rs"` but I'm not sure if that is okay or not.
2 parents 5e3221e + 1b99491 commit 333a798

File tree

2 files changed

+118
-2
lines changed

2 files changed

+118
-2
lines changed

src/cargo/util/toml.rs

+36-2
Original file line numberDiff line numberDiff line change
@@ -286,12 +286,18 @@ pub struct TomlProfile {
286286
panic: Option<String>,
287287
}
288288

289+
#[derive(RustcDecodable, Clone, Debug)]
290+
pub enum StringOrBool {
291+
String(String),
292+
Bool(bool),
293+
}
294+
289295
#[derive(RustcDecodable)]
290296
pub struct TomlProject {
291297
name: String,
292298
version: TomlVersion,
293299
authors: Vec<String>,
294-
build: Option<String>,
300+
build: Option<StringOrBool>,
295301
links: Option<String>,
296302
exclude: Option<Vec<String>>,
297303
include: Option<Vec<String>>,
@@ -540,7 +546,7 @@ impl TomlManifest {
540546
}
541547

542548
// processing the custom build script
543-
let new_build = project.build.as_ref().map(PathBuf::from);
549+
let new_build = self.maybe_custom_build(&project.build, &layout.root, &mut warnings);
544550

545551
// Get targets
546552
let targets = normalize(&lib,
@@ -767,6 +773,34 @@ impl TomlManifest {
767773
}
768774
Ok(replace)
769775
}
776+
777+
fn maybe_custom_build(&self,
778+
build: &Option<StringOrBool>,
779+
project_dir: &Path,
780+
warnings: &mut Vec<String>)
781+
-> Option<PathBuf> {
782+
let build_rs = project_dir.join("build.rs");
783+
match *build {
784+
Some(StringOrBool::Bool(false)) => None, // explicitly no build script
785+
Some(StringOrBool::Bool(true)) => Some(build_rs.into()),
786+
Some(StringOrBool::String(ref s)) => Some(PathBuf::from(s)),
787+
None => {
788+
match fs::metadata(&build_rs) {
789+
// Enable this after the warning has been visible for some time
790+
// Ok(ref e) if e.is_file() => Some(build_rs.into()),
791+
Ok(ref e) if e.is_file() => {
792+
warnings.push("`build.rs` files in the same directory \
793+
as your `Cargo.toml` will soon be treated \
794+
as build scripts. Add `build = false` to \
795+
your `Cargo.toml` to prevent this".into());
796+
None
797+
},
798+
Ok(_) => None,
799+
Err(_) => None,
800+
}
801+
}
802+
}
803+
}
770804
}
771805

772806
/// Will check a list of toml targets, and make sure the target names are unique within a vector.

tests/build-script.rs

+82
Original file line numberDiff line numberDiff line change
@@ -2334,3 +2334,85 @@ fn switch_features_rerun() {
23342334
assert_that(build.cargo("run").arg("-v").arg("--features=foo"),
23352335
execs().with_status(0).with_stdout("foo\n"));
23362336
}
2337+
2338+
#[test]
2339+
fn assume_build_script_when_build_rs_present() {
2340+
let p = project("builder")
2341+
.file("Cargo.toml", r#"
2342+
[package]
2343+
name = "builder"
2344+
version = "0.0.1"
2345+
authors = []
2346+
"#)
2347+
.file("src/main.rs", r#"
2348+
use std::path::Path;
2349+
fn main() {
2350+
let f = env!("OUT_DIR");
2351+
assert!(
2352+
! Path::new(f).join("output").exists()
2353+
);
2354+
}
2355+
"#)
2356+
.file("build.rs", r#"
2357+
use std::env;
2358+
use std::fs::File;
2359+
use std::io::Write;
2360+
use std::path::Path;
2361+
2362+
fn main() {
2363+
let out_dir = env::var_os("OUT_DIR").unwrap();
2364+
let out_dir = Path::new(&out_dir).join("output");
2365+
let mut f = File::create(&out_dir).unwrap();
2366+
f.write_all(b"foo").unwrap();
2367+
}
2368+
"#);
2369+
p.build();
2370+
2371+
assert_that(p.cargo("run").arg("-v"),
2372+
execs().with_status(0).with_stderr("\
2373+
warning: `build.rs` files in the same directory as your `Cargo.toml` will soon be treated \
2374+
as build scripts. Add `build = false` to your `Cargo.toml` to prevent this
2375+
Compiling builder v0.0.1 ([..])
2376+
Running [..]
2377+
Finished [..]
2378+
Running [..]
2379+
"));
2380+
}
2381+
2382+
#[test]
2383+
fn if_build_set_to_false_dont_treat_build_rs_as_build_script() {
2384+
let p = project("builder")
2385+
.file("Cargo.toml", r#"
2386+
[package]
2387+
name = "builder"
2388+
version = "0.0.1"
2389+
authors = []
2390+
build = false
2391+
"#)
2392+
.file("src/main.rs", r#"
2393+
use std::path::Path;
2394+
fn main() {
2395+
let f = env!("OUT_DIR");
2396+
assert!(
2397+
! Path::new(f).join("output").exists()
2398+
)
2399+
}
2400+
"#)
2401+
.file("build.rs", r#"
2402+
use std::env;
2403+
use std::fs::File;
2404+
use std::io::Write;
2405+
use std::path::Path;
2406+
2407+
fn main() {
2408+
let out_dir = env::var_os("OUT_DIR").unwrap();
2409+
let out_dir = Path::new(&out_dir).join("output");
2410+
let mut f = File::create(&out_dir).unwrap();
2411+
f.write_all(b"foo").unwrap();
2412+
}
2413+
"#);
2414+
p.build();
2415+
2416+
assert_that(p.cargo("run").arg("-v"),
2417+
execs().with_status(0));
2418+
}

0 commit comments

Comments
 (0)