Skip to content

Commit 191f65f

Browse files
committed
Auto merge of #2610 - japaric:install-from-tempdir, r=alexcrichton
cargo-install: prefer building artifacts in the system temporary directory and each cargo-install instance creates and uses its own build directory. This allows running several cargo-install instances in parallel. If we fail to create a temporary directory for whatever reason fallback to creating and using a target-install directory in the current directory. closes #2606 --- r? @alexcrichton Qs: - Should we preserve the current behavior (`target-install` in `cwd`) as a fallback or remove it and error if we can't create a `TempDir` in `env::temp_dir()`? (we currently error if we can't create `target-install` directory in `cwd`) - Should I add tests for the issues I raised at #2606? If yes, how can I test `cargo-install` parallelism? Lack of "Blocking waiting for file lock on build directory" in the output of the `cargo` commands? or something else?
2 parents 867627c + b484cd3 commit 191f65f

File tree

5 files changed

+66
-3
lines changed

5 files changed

+66
-3
lines changed

Cargo.toml

+1-1
Original file line numberDiff line numberDiff line change
@@ -38,14 +38,14 @@ regex = "0.1"
3838
rustc-serialize = "0.3"
3939
semver = "0.2.2"
4040
tar = "0.4"
41+
tempdir = "0.3"
4142
term = "0.4.4"
4243
time = "0.1"
4344
toml = "0.1"
4445
url = "0.2"
4546
winapi = "0.2"
4647

4748
[dev-dependencies]
48-
tempdir = "0.3"
4949
hamcrest = "0.1"
5050
bufstream = "0.1"
5151
filetime = "0.1"

src/cargo/lib.rs

+1
Original file line numberDiff line numberDiff line change
@@ -19,6 +19,7 @@ extern crate regex;
1919
extern crate rustc_serialize;
2020
extern crate semver;
2121
extern crate tar;
22+
extern crate tempdir;
2223
extern crate term;
2324
extern crate time;
2425
extern crate toml;

src/cargo/ops/cargo_install.rs

+14-1
Original file line numberDiff line numberDiff line change
@@ -7,6 +7,7 @@ use std::io::prelude::*;
77
use std::io::SeekFrom;
88
use std::path::{Path, PathBuf};
99

10+
use tempdir::TempDir;
1011
use toml;
1112

1213
use core::{SourceId, Source, Package, Registry, Dependency, PackageIdSpec};
@@ -79,13 +80,25 @@ pub fn install(root: Option<&str>,
7980
try!(check_overwrites(&dst, &pkg, &opts.filter, &list));
8081
}
8182

83+
let mut td_opt = None;
8284
let target_dir = if source_id.is_path() {
8385
config.target_dir(&pkg)
8486
} else {
85-
Filesystem::new(config.cwd().join("target-install"))
87+
if let Ok(td) = TempDir::new("cargo-install") {
88+
let p = td.path().to_owned();
89+
td_opt = Some(td);
90+
Filesystem::new(p)
91+
} else {
92+
Filesystem::new(config.cwd().join("target-install"))
93+
}
8694
};
8795
config.set_target_dir(target_dir.clone());
8896
let compile = try!(ops::compile_pkg(&pkg, Some(source), opts).chain_error(|| {
97+
if let Some(td) = td_opt.take() {
98+
// preserve the temporary directory, so the user can inspect it
99+
td.into_path();
100+
}
101+
89102
human(format!("failed to compile `{}`, intermediate artifacts can be \
90103
found at `{}`", pkg, target_dir.display()))
91104
}));

tests/test_cargo_concurrent.rs

+35-1
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,4 @@
1-
use std::env;
1+
use std::{env, str};
22
use std::fs::{self, File};
33
use std::io::Write;
44
use std::net::TcpListener;
@@ -15,6 +15,12 @@ use test_cargo_install::{cargo_home, has_installed_exe};
1515

1616
fn setup() {}
1717

18+
fn pkg(name: &str, vers: &str) {
19+
Package::new(name, vers)
20+
.file("src/main.rs", "fn main() {{}}")
21+
.publish();
22+
}
23+
1824
test!(multiple_installs {
1925
let p = project("foo")
2026
.file("a/Cargo.toml", r#"
@@ -52,6 +58,34 @@ test!(multiple_installs {
5258
assert_that(cargo_home(), has_installed_exe("bar"));
5359
});
5460

61+
test!(concurrent_installs {
62+
const LOCKED_BUILD: &'static str = "waiting for file lock on build directory";
63+
64+
pkg("foo", "0.0.1");
65+
pkg("bar", "0.0.1");
66+
67+
let mut a = ::cargo_process().arg("install").arg("foo").build_command();
68+
let mut b = ::cargo_process().arg("install").arg("bar").build_command();
69+
70+
a.stdout(Stdio::piped()).stderr(Stdio::piped());
71+
b.stdout(Stdio::piped()).stderr(Stdio::piped());
72+
73+
let a = a.spawn().unwrap();
74+
let b = b.spawn().unwrap();
75+
let a = thread::spawn(move || a.wait_with_output().unwrap());
76+
let b = b.wait_with_output().unwrap();
77+
let a = a.join().unwrap();
78+
79+
assert!(!str::from_utf8(&a.stderr).unwrap().contains(LOCKED_BUILD));
80+
assert!(!str::from_utf8(&b.stderr).unwrap().contains(LOCKED_BUILD));
81+
82+
assert_that(a, execs().with_status(0));
83+
assert_that(b, execs().with_status(0));
84+
85+
assert_that(cargo_home(), has_installed_exe("foo"));
86+
assert_that(cargo_home(), has_installed_exe("bar"));
87+
});
88+
5589
test!(one_install_should_be_bad {
5690
let p = project("foo")
5791
.file("a/Cargo.toml", r#"

tests/test_cargo_install.rs

+15
Original file line numberDiff line numberDiff line change
@@ -659,3 +659,18 @@ test!(q_silences_warnings {
659659
assert_that(cargo_process("install").arg("-q").arg("--path").arg(p.root()),
660660
execs().with_status(0).with_stderr(""));
661661
});
662+
663+
test!(readonly_dir {
664+
pkg("foo", "0.0.1");
665+
666+
let root = paths::root();
667+
let dir = &root.join("readonly");
668+
fs::create_dir(root.join("readonly")).unwrap();
669+
let mut perms = fs::metadata(dir).unwrap().permissions();
670+
perms.set_readonly(true);
671+
fs::set_permissions(dir, perms).unwrap();
672+
673+
assert_that(cargo_process("install").arg("foo").cwd(dir),
674+
execs().with_status(0));
675+
assert_that(cargo_home(), has_installed_exe("foo"));
676+
});

0 commit comments

Comments
 (0)