| //! Tests for the `cargo install` command. |
| |
| use std::fs::{self, OpenOptions}; |
| use std::io::prelude::*; |
| |
| use cargo_test_support::cross_compile; |
| use cargo_test_support::git; |
| use cargo_test_support::registry::{self, registry_path, registry_url, Package}; |
| use cargo_test_support::{ |
| basic_manifest, cargo_process, no_such_file_err_msg, project, symlink_supported, t, |
| }; |
| |
| use cargo_test_support::install::{ |
| assert_has_installed_exe, assert_has_not_installed_exe, cargo_home, |
| }; |
| use cargo_test_support::paths::{self, CargoPathExt}; |
| use std::env; |
| use std::path::PathBuf; |
| |
| fn pkg(name: &str, vers: &str) { |
| Package::new(name, vers) |
| .file("src/lib.rs", "") |
| .file( |
| "src/main.rs", |
| &format!("extern crate {}; fn main() {{}}", name), |
| ) |
| .publish(); |
| } |
| |
| #[cargo_test] |
| fn simple() { |
| pkg("foo", "0.0.1"); |
| |
| cargo_process("install foo") |
| .with_stderr( |
| "\ |
| [UPDATING] `[..]` index |
| [DOWNLOADING] crates ... |
| [DOWNLOADED] foo v0.0.1 (registry [..]) |
| [INSTALLING] foo v0.0.1 |
| [COMPILING] foo v0.0.1 |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [CWD]/home/.cargo/bin/foo[EXE] |
| [INSTALLED] package `foo v0.0.1` (executable `foo[EXE]`) |
| [WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries |
| ", |
| ) |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| |
| cargo_process("uninstall foo") |
| .with_stderr("[REMOVING] [CWD]/home/.cargo/bin/foo[EXE]") |
| .run(); |
| assert_has_not_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn with_index() { |
| pkg("foo", "0.0.1"); |
| |
| cargo_process("install foo --index") |
| .arg(registry_url().to_string()) |
| .with_stderr(&format!( |
| "\ |
| [UPDATING] `{reg}` index |
| [DOWNLOADING] crates ... |
| [DOWNLOADED] foo v0.0.1 (registry `{reg}`) |
| [INSTALLING] foo v0.0.1 (registry `{reg}`) |
| [COMPILING] foo v0.0.1 (registry `{reg}`) |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [CWD]/home/.cargo/bin/foo[EXE] |
| [INSTALLED] package `foo v0.0.1 (registry `{reg}`)` (executable `foo[EXE]`) |
| [WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries |
| ", |
| reg = registry_path().to_str().unwrap() |
| )) |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| |
| cargo_process("uninstall foo") |
| .with_stderr("[REMOVING] [CWD]/home/.cargo/bin/foo[EXE]") |
| .run(); |
| assert_has_not_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn multiple_pkgs() { |
| pkg("foo", "0.0.1"); |
| pkg("bar", "0.0.2"); |
| |
| cargo_process("install foo bar baz") |
| .with_status(101) |
| .with_stderr( |
| "\ |
| [UPDATING] `[..]` index |
| [DOWNLOADING] crates ... |
| [DOWNLOADED] foo v0.0.1 (registry `[CWD]/registry`) |
| [INSTALLING] foo v0.0.1 |
| [COMPILING] foo v0.0.1 |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [CWD]/home/.cargo/bin/foo[EXE] |
| [INSTALLED] package `foo v0.0.1` (executable `foo[EXE]`) |
| [DOWNLOADING] crates ... |
| [DOWNLOADED] bar v0.0.2 (registry `[CWD]/registry`) |
| [INSTALLING] bar v0.0.2 |
| [COMPILING] bar v0.0.2 |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [CWD]/home/.cargo/bin/bar[EXE] |
| [INSTALLED] package `bar v0.0.2` (executable `bar[EXE]`) |
| [ERROR] could not find `baz` in registry `[..]` with version `*` |
| [SUMMARY] Successfully installed foo, bar! Failed to install baz (see error(s) above). |
| [WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries |
| [ERROR] some crates failed to install |
| ", |
| ) |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| assert_has_installed_exe(cargo_home(), "bar"); |
| |
| cargo_process("uninstall foo bar") |
| .with_stderr( |
| "\ |
| [REMOVING] [CWD]/home/.cargo/bin/foo[EXE] |
| [REMOVING] [CWD]/home/.cargo/bin/bar[EXE] |
| [SUMMARY] Successfully uninstalled foo, bar! |
| ", |
| ) |
| .run(); |
| |
| assert_has_not_installed_exe(cargo_home(), "foo"); |
| assert_has_not_installed_exe(cargo_home(), "bar"); |
| } |
| |
| fn path() -> Vec<PathBuf> { |
| env::split_paths(&env::var_os("PATH").unwrap_or_default()).collect() |
| } |
| |
| #[cargo_test] |
| fn multiple_pkgs_path_set() { |
| // confirm partial failure results in 101 status code and does not have the |
| // '[WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries' |
| // even if CARGO_HOME/bin is in the PATH |
| pkg("foo", "0.0.1"); |
| pkg("bar", "0.0.2"); |
| |
| // add CARGO_HOME/bin to path |
| let mut path = path(); |
| path.push(cargo_home().join("bin")); |
| let new_path = env::join_paths(path).unwrap(); |
| cargo_process("install foo bar baz") |
| .env("PATH", new_path) |
| .with_status(101) |
| .with_stderr( |
| "\ |
| [UPDATING] `[..]` index |
| [DOWNLOADING] crates ... |
| [DOWNLOADED] foo v0.0.1 (registry `[CWD]/registry`) |
| [INSTALLING] foo v0.0.1 |
| [COMPILING] foo v0.0.1 |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [CWD]/home/.cargo/bin/foo[EXE] |
| [INSTALLED] package `foo v0.0.1` (executable `foo[EXE]`) |
| [DOWNLOADING] crates ... |
| [DOWNLOADED] bar v0.0.2 (registry `[CWD]/registry`) |
| [INSTALLING] bar v0.0.2 |
| [COMPILING] bar v0.0.2 |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [CWD]/home/.cargo/bin/bar[EXE] |
| [INSTALLED] package `bar v0.0.2` (executable `bar[EXE]`) |
| [ERROR] could not find `baz` in registry `[..]` with version `*` |
| [SUMMARY] Successfully installed foo, bar! Failed to install baz (see error(s) above). |
| [ERROR] some crates failed to install |
| ", |
| ) |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| assert_has_installed_exe(cargo_home(), "bar"); |
| |
| cargo_process("uninstall foo bar") |
| .with_stderr( |
| "\ |
| [REMOVING] [CWD]/home/.cargo/bin/foo[EXE] |
| [REMOVING] [CWD]/home/.cargo/bin/bar[EXE] |
| [SUMMARY] Successfully uninstalled foo, bar! |
| ", |
| ) |
| .run(); |
| |
| assert_has_not_installed_exe(cargo_home(), "foo"); |
| assert_has_not_installed_exe(cargo_home(), "bar"); |
| } |
| |
| #[cargo_test] |
| fn pick_max_version() { |
| pkg("foo", "0.1.0"); |
| pkg("foo", "0.2.0"); |
| pkg("foo", "0.2.1"); |
| pkg("foo", "0.2.1-pre.1"); |
| pkg("foo", "0.3.0-pre.2"); |
| |
| cargo_process("install foo") |
| .with_stderr( |
| "\ |
| [UPDATING] `[..]` index |
| [DOWNLOADING] crates ... |
| [DOWNLOADED] foo v0.2.1 (registry [..]) |
| [INSTALLING] foo v0.2.1 |
| [COMPILING] foo v0.2.1 |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [CWD]/home/.cargo/bin/foo[EXE] |
| [INSTALLED] package `foo v0.2.1` (executable `foo[EXE]`) |
| [WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries |
| ", |
| ) |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn installs_beta_version_by_explicit_name_from_git() { |
| let p = git::repo(&paths::root().join("foo")) |
| .file("Cargo.toml", &basic_manifest("foo", "0.3.0-beta.1")) |
| .file("src/main.rs", "fn main() {}") |
| .build(); |
| |
| cargo_process("install --git") |
| .arg(p.url().to_string()) |
| .arg("foo") |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn missing() { |
| pkg("foo", "0.0.1"); |
| cargo_process("install bar") |
| .with_status(101) |
| .with_stderr( |
| "\ |
| [UPDATING] [..] index |
| [ERROR] could not find `bar` in registry `[..]` with version `*` |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn missing_current_working_directory() { |
| cargo_process("install .") |
| .with_status(101) |
| .with_stderr( |
| "\ |
| error: To install the binaries for the package in current working \ |
| directory use `cargo install --path .`. Use `cargo build` if you \ |
| want to simply build the package. |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn bad_version() { |
| pkg("foo", "0.0.1"); |
| cargo_process("install foo --vers=0.2.0") |
| .with_status(101) |
| .with_stderr( |
| "\ |
| [UPDATING] [..] index |
| [ERROR] could not find `foo` in registry `[..]` with version `=0.2.0` |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn bad_paths() { |
| cargo_process("install") |
| .with_status(101) |
| .with_stderr("[ERROR] `[CWD]` is not a crate root; specify a crate to install [..]") |
| .run(); |
| |
| cargo_process("install --path .") |
| .with_status(101) |
| .with_stderr("[ERROR] `[CWD]` does not contain a Cargo.toml file[..]") |
| .run(); |
| |
| let toml = paths::root().join("Cargo.toml"); |
| fs::write(toml, "").unwrap(); |
| cargo_process("install --path Cargo.toml") |
| .with_status(101) |
| .with_stderr("[ERROR] `[CWD]/Cargo.toml` is not a directory[..]") |
| .run(); |
| |
| cargo_process("install --path .") |
| .with_status(101) |
| .with_stderr_contains("[ERROR] failed to parse manifest at `[CWD]/Cargo.toml`") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn install_location_precedence() { |
| pkg("foo", "0.0.1"); |
| |
| let root = paths::root(); |
| let t1 = root.join("t1"); |
| let t2 = root.join("t2"); |
| let t3 = root.join("t3"); |
| let t4 = cargo_home(); |
| |
| fs::create_dir(root.join(".cargo")).unwrap(); |
| fs::write( |
| root.join(".cargo/config"), |
| &format!( |
| "[install] |
| root = '{}' |
| ", |
| t3.display() |
| ), |
| ) |
| .unwrap(); |
| |
| println!("install --root"); |
| |
| cargo_process("install foo --root") |
| .arg(&t1) |
| .env("CARGO_INSTALL_ROOT", &t2) |
| .run(); |
| assert_has_installed_exe(&t1, "foo"); |
| assert_has_not_installed_exe(&t2, "foo"); |
| |
| println!("install CARGO_INSTALL_ROOT"); |
| |
| cargo_process("install foo") |
| .env("CARGO_INSTALL_ROOT", &t2) |
| .run(); |
| assert_has_installed_exe(&t2, "foo"); |
| assert_has_not_installed_exe(&t3, "foo"); |
| |
| println!("install install.root"); |
| |
| cargo_process("install foo").run(); |
| assert_has_installed_exe(&t3, "foo"); |
| assert_has_not_installed_exe(&t4, "foo"); |
| |
| fs::remove_file(root.join(".cargo/config")).unwrap(); |
| |
| println!("install cargo home"); |
| |
| cargo_process("install foo").run(); |
| assert_has_installed_exe(&t4, "foo"); |
| } |
| |
| #[cargo_test] |
| fn install_path() { |
| let p = project().file("src/main.rs", "fn main() {}").build(); |
| |
| cargo_process("install --path").arg(p.root()).run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| // path-style installs force a reinstall |
| p.cargo("install --path .") |
| .with_stderr( |
| "\ |
| [INSTALLING] foo v0.0.1 [..] |
| [FINISHED] release [..] |
| [REPLACING] [..]/.cargo/bin/foo[EXE] |
| [REPLACED] package `foo v0.0.1 [..]` with `foo v0.0.1 [..]` (executable `foo[EXE]`) |
| [WARNING] be sure to add [..] |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn install_target_dir() { |
| let p = project().file("src/main.rs", "fn main() {}").build(); |
| |
| p.cargo("install --target-dir td_test") |
| .with_stderr( |
| "\ |
| [WARNING] Using `cargo install` [..] |
| [INSTALLING] foo v0.0.1 [..] |
| [COMPILING] foo v0.0.1 [..] |
| [FINISHED] release [..] |
| [INSTALLING] [..]foo[EXE] |
| [INSTALLED] package `foo v0.0.1 [..]foo[..]` (executable `foo[EXE]`) |
| [WARNING] be sure to add [..] |
| ", |
| ) |
| .run(); |
| |
| let mut path = p.root(); |
| path.push("td_test"); |
| assert!(path.exists()); |
| |
| #[cfg(not(windows))] |
| path.push("release/foo"); |
| #[cfg(windows)] |
| path.push("release/foo.exe"); |
| assert!(path.exists()); |
| } |
| |
| #[cargo_test] |
| fn multiple_crates_error() { |
| let p = git::repo(&paths::root().join("foo")) |
| .file("Cargo.toml", &basic_manifest("foo", "0.1.0")) |
| .file("src/main.rs", "fn main() {}") |
| .file("a/Cargo.toml", &basic_manifest("bar", "0.1.0")) |
| .file("a/src/main.rs", "fn main() {}") |
| .build(); |
| |
| cargo_process("install --git") |
| .arg(p.url().to_string()) |
| .with_status(101) |
| .with_stderr( |
| "\ |
| [UPDATING] git repository [..] |
| [ERROR] multiple packages with binaries found: bar, foo |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn multiple_crates_select() { |
| let p = git::repo(&paths::root().join("foo")) |
| .file("Cargo.toml", &basic_manifest("foo", "0.1.0")) |
| .file("src/main.rs", "fn main() {}") |
| .file("a/Cargo.toml", &basic_manifest("bar", "0.1.0")) |
| .file("a/src/main.rs", "fn main() {}") |
| .build(); |
| |
| cargo_process("install --git") |
| .arg(p.url().to_string()) |
| .arg("foo") |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| assert_has_not_installed_exe(cargo_home(), "bar"); |
| |
| cargo_process("install --git") |
| .arg(p.url().to_string()) |
| .arg("bar") |
| .run(); |
| assert_has_installed_exe(cargo_home(), "bar"); |
| } |
| |
| #[cargo_test] |
| fn multiple_crates_git_all() { |
| let p = git::repo(&paths::root().join("foo")) |
| .file( |
| "Cargo.toml", |
| r#" |
| [workspace] |
| members = ["bin1", "bin2"] |
| "#, |
| ) |
| .file("bin1/Cargo.toml", &basic_manifest("bin1", "0.1.0")) |
| .file("bin2/Cargo.toml", &basic_manifest("bin2", "0.1.0")) |
| .file( |
| "bin1/src/main.rs", |
| r#"fn main() { println!("Hello, world!"); }"#, |
| ) |
| .file( |
| "bin2/src/main.rs", |
| r#"fn main() { println!("Hello, world!"); }"#, |
| ) |
| .build(); |
| |
| cargo_process(&format!("install --git {} bin1 bin2", p.url().to_string())).run(); |
| } |
| |
| #[cargo_test] |
| fn multiple_crates_auto_binaries() { |
| let p = project() |
| .file( |
| "Cargo.toml", |
| r#" |
| [package] |
| name = "foo" |
| version = "0.1.0" |
| authors = [] |
| |
| [dependencies] |
| bar = { path = "a" } |
| "#, |
| ) |
| .file("src/main.rs", "extern crate bar; fn main() {}") |
| .file("a/Cargo.toml", &basic_manifest("bar", "0.1.0")) |
| .file("a/src/lib.rs", "") |
| .build(); |
| |
| cargo_process("install --path").arg(p.root()).run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn multiple_crates_auto_examples() { |
| let p = project() |
| .file( |
| "Cargo.toml", |
| r#" |
| [package] |
| name = "foo" |
| version = "0.1.0" |
| authors = [] |
| |
| [dependencies] |
| bar = { path = "a" } |
| "#, |
| ) |
| .file("src/lib.rs", "extern crate bar;") |
| .file( |
| "examples/foo.rs", |
| " |
| extern crate bar; |
| extern crate foo; |
| fn main() {} |
| ", |
| ) |
| .file("a/Cargo.toml", &basic_manifest("bar", "0.1.0")) |
| .file("a/src/lib.rs", "") |
| .build(); |
| |
| cargo_process("install --path") |
| .arg(p.root()) |
| .arg("--example=foo") |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn no_binaries_or_examples() { |
| let p = project() |
| .file( |
| "Cargo.toml", |
| r#" |
| [package] |
| name = "foo" |
| version = "0.1.0" |
| authors = [] |
| |
| [dependencies] |
| bar = { path = "a" } |
| "#, |
| ) |
| .file("src/lib.rs", "") |
| .file("a/Cargo.toml", &basic_manifest("bar", "0.1.0")) |
| .file("a/src/lib.rs", "") |
| .build(); |
| |
| cargo_process("install --path") |
| .arg(p.root()) |
| .with_status(101) |
| .with_stderr("[ERROR] no packages found with binaries or examples") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn no_binaries() { |
| let p = project() |
| .file("src/lib.rs", "") |
| .file("examples/foo.rs", "fn main() {}") |
| .build(); |
| |
| cargo_process("install --path") |
| .arg(p.root()) |
| .arg("foo") |
| .with_status(101) |
| .with_stderr( |
| "\ |
| [ERROR] there is nothing to install in `foo v0.0.1 ([..])`, because it has no binaries[..] |
| [..] |
| [..]", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn examples() { |
| let p = project() |
| .file("src/lib.rs", "") |
| .file("examples/foo.rs", "extern crate foo; fn main() {}") |
| .build(); |
| |
| cargo_process("install --path") |
| .arg(p.root()) |
| .arg("--example=foo") |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn install_force() { |
| let p = project().file("src/main.rs", "fn main() {}").build(); |
| |
| cargo_process("install --path").arg(p.root()).run(); |
| |
| let p = project() |
| .at("foo2") |
| .file("Cargo.toml", &basic_manifest("foo", "0.2.0")) |
| .file("src/main.rs", "fn main() {}") |
| .build(); |
| |
| cargo_process("install --force --path") |
| .arg(p.root()) |
| .with_stderr( |
| "\ |
| [INSTALLING] foo v0.2.0 ([..]) |
| [COMPILING] foo v0.2.0 ([..]) |
| [FINISHED] release [optimized] target(s) in [..] |
| [REPLACING] [CWD]/home/.cargo/bin/foo[EXE] |
| [REPLACED] package `foo v0.0.1 ([..]/foo)` with `foo v0.2.0 ([..]/foo2)` (executable `foo[EXE]`) |
| [WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries |
| ", |
| ) |
| .run(); |
| |
| cargo_process("install --list") |
| .with_stdout( |
| "\ |
| foo v0.2.0 ([..]): |
| foo[..] |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn install_force_partial_overlap() { |
| let p = project() |
| .file("src/bin/foo-bin1.rs", "fn main() {}") |
| .file("src/bin/foo-bin2.rs", "fn main() {}") |
| .build(); |
| |
| cargo_process("install --path").arg(p.root()).run(); |
| |
| let p = project() |
| .at("foo2") |
| .file("Cargo.toml", &basic_manifest("foo", "0.2.0")) |
| .file("src/bin/foo-bin2.rs", "fn main() {}") |
| .file("src/bin/foo-bin3.rs", "fn main() {}") |
| .build(); |
| |
| cargo_process("install --force --path") |
| .arg(p.root()) |
| .with_stderr( |
| "\ |
| [INSTALLING] foo v0.2.0 ([..]) |
| [COMPILING] foo v0.2.0 ([..]) |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [CWD]/home/.cargo/bin/foo-bin3[EXE] |
| [REPLACING] [CWD]/home/.cargo/bin/foo-bin2[EXE] |
| [REMOVING] executable `[..]/bin/foo-bin1[EXE]` from previous version foo v0.0.1 [..] |
| [INSTALLED] package `foo v0.2.0 ([..]/foo2)` (executable `foo-bin3[EXE]`) |
| [REPLACED] package `foo v0.0.1 ([..]/foo)` with `foo v0.2.0 ([..]/foo2)` (executable `foo-bin2[EXE]`) |
| [WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries |
| ", |
| ) |
| .run(); |
| |
| cargo_process("install --list") |
| .with_stdout( |
| "\ |
| foo v0.2.0 ([..]): |
| foo-bin2[..] |
| foo-bin3[..] |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn install_force_bin() { |
| let p = project() |
| .file("src/bin/foo-bin1.rs", "fn main() {}") |
| .file("src/bin/foo-bin2.rs", "fn main() {}") |
| .build(); |
| |
| cargo_process("install --path").arg(p.root()).run(); |
| |
| let p = project() |
| .at("foo2") |
| .file("Cargo.toml", &basic_manifest("foo", "0.2.0")) |
| .file("src/bin/foo-bin1.rs", "fn main() {}") |
| .file("src/bin/foo-bin2.rs", "fn main() {}") |
| .build(); |
| |
| cargo_process("install --force --bin foo-bin2 --path") |
| .arg(p.root()) |
| .with_stderr( |
| "\ |
| [INSTALLING] foo v0.2.0 ([..]) |
| [COMPILING] foo v0.2.0 ([..]) |
| [FINISHED] release [optimized] target(s) in [..] |
| [REPLACING] [CWD]/home/.cargo/bin/foo-bin2[EXE] |
| [REPLACED] package `foo v0.0.1 ([..]/foo)` with `foo v0.2.0 ([..]/foo2)` (executable `foo-bin2[EXE]`) |
| [WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries |
| ", |
| ) |
| .run(); |
| |
| cargo_process("install --list") |
| .with_stdout( |
| "\ |
| foo v0.0.1 ([..]): |
| foo-bin1[..] |
| foo v0.2.0 ([..]): |
| foo-bin2[..] |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn compile_failure() { |
| let p = project().file("src/main.rs", "").build(); |
| |
| cargo_process("install --path") |
| .arg(p.root()) |
| .with_status(101) |
| .with_stderr_contains( |
| "\ |
| [ERROR] failed to compile `foo v0.0.1 ([..])`, intermediate artifacts can be \ |
| found at `[..]target` |
| |
| Caused by: |
| could not compile `foo` |
| |
| To learn more, run the command again with --verbose. |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn git_repo() { |
| let p = git::repo(&paths::root().join("foo")) |
| .file("Cargo.toml", &basic_manifest("foo", "0.1.0")) |
| .file("src/main.rs", "fn main() {}") |
| .build(); |
| |
| // Use `--locked` to test that we don't even try to write a lock file. |
| cargo_process("install --locked --git") |
| .arg(p.url().to_string()) |
| .with_stderr( |
| "\ |
| [UPDATING] git repository `[..]` |
| [WARNING] no Cargo.lock file published in foo v0.1.0 ([..]) |
| [INSTALLING] foo v0.1.0 ([..]) |
| [COMPILING] foo v0.1.0 ([..]) |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [CWD]/home/.cargo/bin/foo[EXE] |
| [INSTALLED] package `foo v0.1.0 ([..]/foo#[..])` (executable `foo[EXE]`) |
| [WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries |
| ", |
| ) |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn list() { |
| pkg("foo", "0.0.1"); |
| pkg("bar", "0.2.1"); |
| pkg("bar", "0.2.2"); |
| |
| cargo_process("install --list").with_stdout("").run(); |
| |
| cargo_process("install bar --vers =0.2.1").run(); |
| cargo_process("install foo").run(); |
| cargo_process("install --list") |
| .with_stdout( |
| "\ |
| bar v0.2.1: |
| bar[..] |
| foo v0.0.1: |
| foo[..] |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn list_error() { |
| pkg("foo", "0.0.1"); |
| cargo_process("install foo").run(); |
| cargo_process("install --list") |
| .with_stdout( |
| "\ |
| foo v0.0.1: |
| foo[..] |
| ", |
| ) |
| .run(); |
| let mut worldfile_path = cargo_home(); |
| worldfile_path.push(".crates.toml"); |
| let mut worldfile = OpenOptions::new() |
| .write(true) |
| .open(worldfile_path) |
| .expect(".crates.toml should be there"); |
| worldfile.write_all(b"\x00").unwrap(); |
| drop(worldfile); |
| cargo_process("install --list --verbose") |
| .with_status(101) |
| .with_stderr( |
| "\ |
| [ERROR] failed to parse crate metadata at `[..]` |
| |
| Caused by: |
| invalid TOML found for metadata |
| |
| Caused by: |
| unexpected character[..] |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn uninstall_pkg_does_not_exist() { |
| cargo_process("uninstall foo") |
| .with_status(101) |
| .with_stderr("[ERROR] package ID specification `foo` did not match any packages") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn uninstall_bin_does_not_exist() { |
| pkg("foo", "0.0.1"); |
| |
| cargo_process("install foo").run(); |
| cargo_process("uninstall foo --bin=bar") |
| .with_status(101) |
| .with_stderr("[ERROR] binary `bar[..]` not installed as part of `foo v0.0.1`") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn uninstall_piecemeal() { |
| let p = project() |
| .file("src/bin/foo.rs", "fn main() {}") |
| .file("src/bin/bar.rs", "fn main() {}") |
| .build(); |
| |
| cargo_process("install --path").arg(p.root()).run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| assert_has_installed_exe(cargo_home(), "bar"); |
| |
| cargo_process("uninstall foo --bin=bar") |
| .with_stderr("[REMOVING] [..]bar[..]") |
| .run(); |
| |
| assert_has_installed_exe(cargo_home(), "foo"); |
| assert_has_not_installed_exe(cargo_home(), "bar"); |
| |
| cargo_process("uninstall foo --bin=foo") |
| .with_stderr("[REMOVING] [..]foo[..]") |
| .run(); |
| assert_has_not_installed_exe(cargo_home(), "foo"); |
| |
| cargo_process("uninstall foo") |
| .with_status(101) |
| .with_stderr("[ERROR] package ID specification `foo` did not match any packages") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn subcommand_works_out_of_the_box() { |
| Package::new("cargo-foo", "1.0.0") |
| .file("src/main.rs", r#"fn main() { println!("bar"); }"#) |
| .publish(); |
| cargo_process("install cargo-foo").run(); |
| cargo_process("foo").with_stdout("bar\n").run(); |
| cargo_process("--list") |
| .with_stdout_contains(" foo\n") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn installs_from_cwd_by_default() { |
| let p = project().file("src/main.rs", "fn main() {}").build(); |
| |
| p.cargo("install") |
| .with_stderr_contains( |
| "warning: Using `cargo install` to install the binaries for the \ |
| package in current working directory is deprecated, \ |
| use `cargo install --path .` instead. \ |
| Use `cargo build` if you want to simply build the package.", |
| ) |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn installs_from_cwd_with_2018_warnings() { |
| let p = project() |
| .file( |
| "Cargo.toml", |
| r#" |
| [package] |
| name = "foo" |
| version = "0.1.0" |
| authors = [] |
| edition = "2018" |
| "#, |
| ) |
| .file("src/main.rs", "fn main() {}") |
| .build(); |
| |
| p.cargo("install") |
| .with_status(101) |
| .with_stderr_contains( |
| "error: Using `cargo install` to install the binaries for the \ |
| package in current working directory is no longer supported, \ |
| use `cargo install --path .` instead. \ |
| Use `cargo build` if you want to simply build the package.", |
| ) |
| .run(); |
| assert_has_not_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn uninstall_cwd() { |
| let p = project().file("src/main.rs", "fn main() {}").build(); |
| p.cargo("install --path .") |
| .with_stderr(&format!( |
| "\ |
| [INSTALLING] foo v0.0.1 ([CWD]) |
| [COMPILING] foo v0.0.1 ([CWD]) |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] {home}/bin/foo[EXE] |
| [INSTALLED] package `foo v0.0.1 ([..]/foo)` (executable `foo[EXE]`) |
| [WARNING] be sure to add `{home}/bin` to your PATH to be able to run the installed binaries", |
| home = cargo_home().display(), |
| )) |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| |
| p.cargo("uninstall") |
| .with_stdout("") |
| .with_stderr(&format!( |
| "[REMOVING] {home}/bin/foo[EXE]", |
| home = cargo_home().display() |
| )) |
| .run(); |
| assert_has_not_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn uninstall_cwd_not_installed() { |
| let p = project().file("src/main.rs", "fn main() {}").build(); |
| p.cargo("uninstall") |
| .with_status(101) |
| .with_stdout("") |
| .with_stderr("error: package `foo v0.0.1 ([CWD])` is not installed") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn uninstall_cwd_no_project() { |
| cargo_process("uninstall") |
| .with_status(101) |
| .with_stdout("") |
| .with_stderr(format!( |
| "\ |
| [ERROR] failed to read `[CWD]/Cargo.toml` |
| |
| Caused by: |
| {err_msg}", |
| err_msg = no_such_file_err_msg(), |
| )) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn do_not_rebuilds_on_local_install() { |
| let p = project().file("src/main.rs", "fn main() {}").build(); |
| |
| p.cargo("build --release").run(); |
| cargo_process("install --path") |
| .arg(p.root()) |
| .with_stderr( |
| "\ |
| [INSTALLING] [..] |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [..] |
| [INSTALLED] package `foo v0.0.1 ([..]/foo)` (executable `foo[EXE]`) |
| [WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries |
| ", |
| ) |
| .run(); |
| |
| assert!(p.build_dir().exists()); |
| assert!(p.release_bin("foo").exists()); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn reports_unsuccessful_subcommand_result() { |
| Package::new("cargo-fail", "1.0.0") |
| .file("src/main.rs", "fn main() { panic!(); }") |
| .publish(); |
| cargo_process("install cargo-fail").run(); |
| cargo_process("--list") |
| .with_stdout_contains(" fail\n") |
| .run(); |
| cargo_process("fail") |
| .with_status(101) |
| .with_stderr_contains("thread '[..]' panicked at 'explicit panic', [..]") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn git_with_lockfile() { |
| let p = git::repo(&paths::root().join("foo")) |
| .file( |
| "Cargo.toml", |
| r#" |
| [package] |
| name = "foo" |
| version = "0.1.0" |
| authors = [] |
| |
| [dependencies] |
| bar = { path = "bar" } |
| "#, |
| ) |
| .file("src/main.rs", "fn main() {}") |
| .file("bar/Cargo.toml", &basic_manifest("bar", "0.1.0")) |
| .file("bar/src/lib.rs", "fn main() {}") |
| .file( |
| "Cargo.lock", |
| r#" |
| [[package]] |
| name = "foo" |
| version = "0.1.0" |
| dependencies = [ "bar 0.1.0" ] |
| |
| [[package]] |
| name = "bar" |
| version = "0.1.0" |
| "#, |
| ) |
| .build(); |
| |
| cargo_process("install --git") |
| .arg(p.url().to_string()) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn q_silences_warnings() { |
| let p = project().file("src/main.rs", "fn main() {}").build(); |
| |
| cargo_process("install -q --path") |
| .arg(p.root()) |
| .with_stderr("") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn readonly_dir() { |
| pkg("foo", "0.0.1"); |
| |
| let root = paths::root(); |
| let dir = &root.join("readonly"); |
| fs::create_dir(root.join("readonly")).unwrap(); |
| let mut perms = fs::metadata(dir).unwrap().permissions(); |
| perms.set_readonly(true); |
| fs::set_permissions(dir, perms).unwrap(); |
| |
| cargo_process("install foo").cwd(dir).run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn use_path_workspace() { |
| Package::new("foo", "1.0.0").publish(); |
| let p = project() |
| .file( |
| "Cargo.toml", |
| r#" |
| [package] |
| name = "bar" |
| version = "0.1.0" |
| authors = [] |
| |
| [workspace] |
| members = ["baz"] |
| "#, |
| ) |
| .file("src/main.rs", "fn main() {}") |
| .file( |
| "baz/Cargo.toml", |
| r#" |
| [package] |
| name = "baz" |
| version = "0.1.0" |
| authors = [] |
| |
| [dependencies] |
| foo = "1" |
| "#, |
| ) |
| .file("baz/src/lib.rs", "") |
| .build(); |
| |
| p.cargo("build").run(); |
| let lock = p.read_lockfile(); |
| p.cargo("install").run(); |
| let lock2 = p.read_lockfile(); |
| assert_eq!(lock, lock2, "different lockfiles"); |
| } |
| |
| #[cargo_test] |
| fn dev_dependencies_no_check() { |
| Package::new("foo", "1.0.0").publish(); |
| let p = project() |
| .file( |
| "Cargo.toml", |
| r#" |
| [package] |
| name = "bar" |
| version = "0.1.0" |
| authors = [] |
| |
| [dev-dependencies] |
| baz = "1.0.0" |
| "#, |
| ) |
| .file("src/main.rs", "fn main() {}") |
| .build(); |
| |
| p.cargo("build") |
| .with_status(101) |
| .with_stderr_contains("[..] no matching package named `baz` found") |
| .run(); |
| p.cargo("install").run(); |
| } |
| |
| #[cargo_test] |
| fn dev_dependencies_lock_file_untouched() { |
| Package::new("foo", "1.0.0").publish(); |
| let p = project() |
| .file( |
| "Cargo.toml", |
| r#" |
| [package] |
| name = "foo" |
| version = "0.1.0" |
| authors = [] |
| |
| [dev-dependencies] |
| bar = { path = "a" } |
| "#, |
| ) |
| .file("src/main.rs", "fn main() {}") |
| .file("a/Cargo.toml", &basic_manifest("bar", "0.1.0")) |
| .file("a/src/lib.rs", "") |
| .build(); |
| |
| p.cargo("build").run(); |
| let lock = p.read_lockfile(); |
| p.cargo("install").run(); |
| let lock2 = p.read_lockfile(); |
| assert!(lock == lock2, "different lockfiles"); |
| } |
| |
| #[cargo_test] |
| fn install_target_native() { |
| pkg("foo", "0.1.0"); |
| |
| cargo_process("install foo --target") |
| .arg(cargo_test_support::rustc_host()) |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn install_target_foreign() { |
| if cross_compile::disabled() { |
| return; |
| } |
| |
| pkg("foo", "0.1.0"); |
| |
| cargo_process("install foo --target") |
| .arg(cross_compile::alternate()) |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn vers_precise() { |
| pkg("foo", "0.1.1"); |
| pkg("foo", "0.1.2"); |
| |
| cargo_process("install foo --vers 0.1.1") |
| .with_stderr_contains("[DOWNLOADED] foo v0.1.1 (registry [..])") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn version_too() { |
| pkg("foo", "0.1.1"); |
| pkg("foo", "0.1.2"); |
| |
| cargo_process("install foo --version 0.1.1") |
| .with_stderr_contains("[DOWNLOADED] foo v0.1.1 (registry [..])") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn not_both_vers_and_version() { |
| pkg("foo", "0.1.1"); |
| pkg("foo", "0.1.2"); |
| |
| cargo_process("install foo --version 0.1.1 --vers 0.1.2") |
| .with_status(1) |
| .with_stderr_contains( |
| "\ |
| error: The argument '--version <VERSION>' was provided more than once, \ |
| but cannot be used multiple times |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn test_install_git_cannot_be_a_base_url() { |
| cargo_process("install --git github.com:rust-lang-nursery/rustfmt.git") |
| .with_status(101) |
| .with_stderr("\ |
| [ERROR] invalid url `github.com:rust-lang-nursery/rustfmt.git`: cannot-be-a-base-URLs are not supported") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn uninstall_multiple_and_specifying_bin() { |
| cargo_process("uninstall foo bar --bin baz") |
| .with_status(101) |
| .with_stderr("\ |
| [ERROR] A binary can only be associated with a single installed package, specifying multiple specs with --bin is redundant.") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn uninstall_with_empty_pakcage_option() { |
| cargo_process("uninstall -p") |
| .with_status(101) |
| .with_stderr( |
| "\ |
| [ERROR] \"--package <SPEC>\" requires a SPEC format value. |
| Run `cargo help pkgid` for more information about SPEC format. |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn uninstall_multiple_and_some_pkg_does_not_exist() { |
| pkg("foo", "0.0.1"); |
| |
| cargo_process("install foo").run(); |
| |
| cargo_process("uninstall foo bar") |
| .with_status(101) |
| .with_stderr( |
| "\ |
| [REMOVING] [CWD]/home/.cargo/bin/foo[EXE] |
| error: package ID specification `bar` did not match any packages |
| [SUMMARY] Successfully uninstalled foo! Failed to uninstall bar (see error(s) above). |
| error: some packages failed to uninstall |
| ", |
| ) |
| .run(); |
| |
| assert_has_not_installed_exe(cargo_home(), "foo"); |
| assert_has_not_installed_exe(cargo_home(), "bar"); |
| } |
| |
| #[cargo_test] |
| fn custom_target_dir_for_git_source() { |
| let p = git::repo(&paths::root().join("foo")) |
| .file("Cargo.toml", &basic_manifest("foo", "0.1.0")) |
| .file("src/main.rs", "fn main() {}") |
| .build(); |
| |
| cargo_process("install --git") |
| .arg(p.url().to_string()) |
| .run(); |
| assert!(!paths::root().join("target/release").is_dir()); |
| |
| cargo_process("install --force --git") |
| .arg(p.url().to_string()) |
| .env("CARGO_TARGET_DIR", "target") |
| .run(); |
| assert!(paths::root().join("target/release").is_dir()); |
| } |
| |
| #[cargo_test] |
| fn install_respects_lock_file() { |
| // `cargo install` now requires --locked to use a Cargo.lock. |
| Package::new("bar", "0.1.0").publish(); |
| Package::new("bar", "0.1.1") |
| .file("src/lib.rs", "not rust") |
| .publish(); |
| Package::new("foo", "0.1.0") |
| .dep("bar", "0.1") |
| .file("src/lib.rs", "") |
| .file( |
| "src/main.rs", |
| "extern crate foo; extern crate bar; fn main() {}", |
| ) |
| .file( |
| "Cargo.lock", |
| r#" |
| [[package]] |
| name = "bar" |
| version = "0.1.0" |
| source = "registry+https://github.com/rust-lang/crates.io-index" |
| |
| [[package]] |
| name = "foo" |
| version = "0.1.0" |
| dependencies = [ |
| "bar 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", |
| ] |
| "#, |
| ) |
| .publish(); |
| |
| cargo_process("install foo") |
| .with_stderr_contains("[..]not rust[..]") |
| .with_status(101) |
| .run(); |
| cargo_process("install --locked foo").run(); |
| } |
| |
| #[cargo_test] |
| fn install_path_respects_lock_file() { |
| // --path version of install_path_respects_lock_file, --locked is required |
| // to use Cargo.lock. |
| Package::new("bar", "0.1.0").publish(); |
| Package::new("bar", "0.1.1") |
| .file("src/lib.rs", "not rust") |
| .publish(); |
| let p = project() |
| .file( |
| "Cargo.toml", |
| r#" |
| [package] |
| name = "foo" |
| version = "0.1.0" |
| |
| [dependencies] |
| bar = "0.1" |
| "#, |
| ) |
| .file("src/main.rs", "extern crate bar; fn main() {}") |
| .file( |
| "Cargo.lock", |
| r#" |
| [[package]] |
| name = "bar" |
| version = "0.1.0" |
| source = "registry+https://github.com/rust-lang/crates.io-index" |
| |
| [[package]] |
| name = "foo" |
| version = "0.1.0" |
| dependencies = [ |
| "bar 0.1.0 (registry+https://github.com/rust-lang/crates.io-index)", |
| ] |
| "#, |
| ) |
| .build(); |
| |
| p.cargo("install --path .") |
| .with_stderr_contains("[..]not rust[..]") |
| .with_status(101) |
| .run(); |
| p.cargo("install --path . --locked").run(); |
| } |
| |
| #[cargo_test] |
| fn lock_file_path_deps_ok() { |
| Package::new("bar", "0.1.0").publish(); |
| |
| Package::new("foo", "0.1.0") |
| .dep("bar", "0.1") |
| .file("src/lib.rs", "") |
| .file( |
| "src/main.rs", |
| "extern crate foo; extern crate bar; fn main() {}", |
| ) |
| .file( |
| "Cargo.lock", |
| r#" |
| [[package]] |
| name = "bar" |
| version = "0.1.0" |
| |
| [[package]] |
| name = "foo" |
| version = "0.1.0" |
| dependencies = [ |
| "bar 0.1.0", |
| ] |
| "#, |
| ) |
| .publish(); |
| |
| cargo_process("install foo").run(); |
| } |
| |
| #[cargo_test] |
| fn install_empty_argument() { |
| // Bug 5229 |
| cargo_process("install") |
| .arg("") |
| .with_status(1) |
| .with_stderr_contains( |
| "[ERROR] The argument '<crate>...' requires a value but none was supplied", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn git_repo_replace() { |
| let p = git::repo(&paths::root().join("foo")) |
| .file("Cargo.toml", &basic_manifest("foo", "0.1.0")) |
| .file("src/main.rs", "fn main() {}") |
| .build(); |
| let repo = git2::Repository::open(&p.root()).unwrap(); |
| let old_rev = repo.revparse_single("HEAD").unwrap().id(); |
| cargo_process("install --git") |
| .arg(p.url().to_string()) |
| .run(); |
| git::commit(&repo); |
| let new_rev = repo.revparse_single("HEAD").unwrap().id(); |
| let mut path = paths::home(); |
| path.push(".cargo/.crates.toml"); |
| |
| assert_ne!(old_rev, new_rev); |
| assert!(fs::read_to_string(path.clone()) |
| .unwrap() |
| .contains(&format!("{}", old_rev))); |
| cargo_process("install --force --git") |
| .arg(p.url().to_string()) |
| .run(); |
| assert!(fs::read_to_string(path) |
| .unwrap() |
| .contains(&format!("{}", new_rev))); |
| } |
| |
| #[cargo_test] |
| fn workspace_uses_workspace_target_dir() { |
| let p = project() |
| .file( |
| "Cargo.toml", |
| r#" |
| [package] |
| name = "foo" |
| version = "0.1.0" |
| authors = [] |
| |
| [workspace] |
| |
| [dependencies] |
| bar = { path = 'bar' } |
| "#, |
| ) |
| .file("src/main.rs", "fn main() {}") |
| .file("bar/Cargo.toml", &basic_manifest("bar", "0.1.0")) |
| .file("bar/src/main.rs", "fn main() {}") |
| .build(); |
| |
| p.cargo("build --release").cwd("bar").run(); |
| cargo_process("install --path") |
| .arg(p.root().join("bar")) |
| .with_stderr( |
| "[INSTALLING] [..] |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [..] |
| [INSTALLED] package `bar v0.1.0 ([..]/bar)` (executable `bar[EXE]`) |
| [WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn install_ignores_local_cargo_config() { |
| pkg("bar", "0.0.1"); |
| |
| let p = project() |
| .file( |
| ".cargo/config", |
| r#" |
| [build] |
| target = "non-existing-target" |
| "#, |
| ) |
| .file("src/main.rs", "fn main() {}") |
| .build(); |
| |
| p.cargo("install bar").run(); |
| assert_has_installed_exe(cargo_home(), "bar"); |
| } |
| |
| #[cargo_test] |
| fn install_ignores_unstable_table_in_local_cargo_config() { |
| pkg("bar", "0.0.1"); |
| |
| let p = project() |
| .file( |
| ".cargo/config", |
| r#" |
| [unstable] |
| build-std = ["core"] |
| "#, |
| ) |
| .file("src/main.rs", "fn main() {}") |
| .build(); |
| |
| p.cargo("install bar").masquerade_as_nightly_cargo().run(); |
| assert_has_installed_exe(cargo_home(), "bar"); |
| } |
| |
| #[cargo_test] |
| fn install_global_cargo_config() { |
| pkg("bar", "0.0.1"); |
| |
| let config = cargo_home().join("config"); |
| let mut toml = fs::read_to_string(&config).unwrap_or_default(); |
| |
| toml.push_str( |
| r#" |
| [build] |
| target = 'nonexistent' |
| "#, |
| ); |
| fs::write(&config, toml).unwrap(); |
| |
| cargo_process("install bar") |
| .with_status(101) |
| .with_stderr_contains("[..]--target nonexistent[..]") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn install_path_config() { |
| project() |
| .file( |
| ".cargo/config", |
| r#" |
| [build] |
| target = 'nonexistent' |
| "#, |
| ) |
| .file("src/main.rs", "fn main() {}") |
| .build(); |
| cargo_process("install --path foo") |
| .with_status(101) |
| .with_stderr_contains("[..]--target nonexistent[..]") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn install_version_req() { |
| // Try using a few versionreq styles. |
| pkg("foo", "0.0.3"); |
| pkg("foo", "1.0.4"); |
| pkg("foo", "1.0.5"); |
| cargo_process("install foo --version=*") |
| .with_stderr_does_not_contain("[WARNING][..]is not a valid semver[..]") |
| .with_stderr_contains("[INSTALLING] foo v1.0.5") |
| .run(); |
| cargo_process("uninstall foo").run(); |
| cargo_process("install foo --version=^1.0") |
| .with_stderr_does_not_contain("[WARNING][..]is not a valid semver[..]") |
| .with_stderr_contains("[INSTALLING] foo v1.0.5") |
| .run(); |
| cargo_process("uninstall foo").run(); |
| cargo_process("install foo --version=0.0.*") |
| .with_stderr_does_not_contain("[WARNING][..]is not a valid semver[..]") |
| .with_stderr_contains("[INSTALLING] foo v0.0.3") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn git_install_reads_workspace_manifest() { |
| let p = git::repo(&paths::root().join("foo")) |
| .file( |
| "Cargo.toml", |
| r#" |
| [workspace] |
| members = ["bin1"] |
| |
| [profile.release] |
| incremental = 3 |
| "#, |
| ) |
| .file("bin1/Cargo.toml", &basic_manifest("bin1", "0.1.0")) |
| .file( |
| "bin1/src/main.rs", |
| r#"fn main() { println!("Hello, world!"); }"#, |
| ) |
| .build(); |
| |
| cargo_process(&format!("install --git {}", p.url().to_string())) |
| .with_status(101) |
| .with_stderr_contains(" invalid type: integer `3`[..]") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn install_git_with_symlink_home() { |
| // Ensure that `cargo install` with a git repo is OK when CARGO_HOME is a |
| // symlink, and uses an build script. |
| if !symlink_supported() { |
| return; |
| } |
| let p = git::new("foo", |p| { |
| p.file("Cargo.toml", &basic_manifest("foo", "1.0.0")) |
| .file("src/main.rs", "fn main() {}") |
| // This triggers discover_git_and_list_files for detecting changed files. |
| .file("build.rs", "fn main() {}") |
| }); |
| #[cfg(unix)] |
| use std::os::unix::fs::symlink; |
| #[cfg(windows)] |
| use std::os::windows::fs::symlink_dir as symlink; |
| |
| let actual = paths::root().join("actual-home"); |
| t!(std::fs::create_dir(&actual)); |
| t!(symlink(&actual, paths::home().join(".cargo"))); |
| cargo_process("install --git") |
| .arg(p.url().to_string()) |
| .with_stderr( |
| "\ |
| [UPDATING] git repository [..] |
| [INSTALLING] foo v1.0.0 [..] |
| [COMPILING] foo v1.0.0 [..] |
| [FINISHED] [..] |
| [INSTALLING] [..]home/.cargo/bin/foo[..] |
| [INSTALLED] package `foo [..] |
| [WARNING] be sure to add [..] |
| ", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn install_yanked_cargo_package() { |
| Package::new("baz", "0.0.1").yanked(true).publish(); |
| cargo_process("install baz --version 0.0.1") |
| .with_status(101) |
| .with_stderr_contains( |
| "error: cannot install package `baz`, it has been yanked from registry \ |
| `https://github.com/rust-lang/crates.io-index`", |
| ) |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn install_cargo_package_in_a_patched_workspace() { |
| pkg("foo", "0.1.0"); |
| pkg("fizz", "1.0.0"); |
| |
| let p = project() |
| .file( |
| "Cargo.toml", |
| r#" |
| [package] |
| name = "bar" |
| version = "0.1.0" |
| authors = [] |
| |
| [workspace] |
| members = ["baz"] |
| "#, |
| ) |
| .file("src/main.rs", "fn main() {}") |
| .file( |
| "baz/Cargo.toml", |
| r#" |
| [package] |
| name = "baz" |
| version = "0.1.0" |
| authors = [] |
| |
| [dependencies] |
| fizz = "1" |
| |
| [patch.crates-io] |
| fizz = { version = "=1.0.0" } |
| "#, |
| ) |
| .file("baz/src/lib.rs", "") |
| .build(); |
| |
| let stderr = "\ |
| [WARNING] patch for the non root package will be ignored, specify patch at the workspace root: |
| package: [..]/foo/baz/Cargo.toml |
| workspace: [..]/foo/Cargo.toml |
| "; |
| p.cargo("check").with_stderr_contains(&stderr).run(); |
| |
| // A crate installation must not emit any message from a workspace under |
| // current working directory. |
| // See https://github.com/rust-lang/cargo/issues/8619 |
| p.cargo("install foo") |
| .with_stderr( |
| "\ |
| [UPDATING] `[..]` index |
| [DOWNLOADING] crates ... |
| [DOWNLOADED] foo v0.1.0 (registry [..]) |
| [INSTALLING] foo v0.1.0 |
| [COMPILING] foo v0.1.0 |
| [FINISHED] release [optimized] target(s) in [..] |
| [INSTALLING] [..]foo[EXE] |
| [INSTALLED] package `foo v0.1.0` (executable `foo[EXE]`) |
| [WARNING] be sure to add `[..]` to your PATH to be able to run the installed binaries |
| ", |
| ) |
| .run(); |
| assert_has_installed_exe(cargo_home(), "foo"); |
| } |
| |
| #[cargo_test] |
| fn locked_install_without_published_lockfile() { |
| Package::new("foo", "0.1.0") |
| .file("src/main.rs", "//! Some docs\nfn main() {}") |
| .publish(); |
| |
| cargo_process("install foo --locked") |
| .with_stderr_contains("[WARNING] no Cargo.lock file published in foo v0.1.0") |
| .run(); |
| } |
| |
| #[cargo_test] |
| fn install_semver_metadata() { |
| // Check trying to install a package that uses semver metadata. |
| // This uses alt registry because the bug this is exercising doesn't |
| // trigger with a replaced source. |
| registry::alt_init(); |
| Package::new("foo", "1.0.0+abc") |
| .alternative(true) |
| .file("src/main.rs", "fn main() {}") |
| .publish(); |
| |
| cargo_process("install foo --registry alternative --version 1.0.0+abc").run(); |
| cargo_process("install foo --registry alternative") |
| .with_stderr("\ |
| [UPDATING] `[ROOT]/alternative-registry` index |
| [IGNORED] package `foo v1.0.0+abc (registry `[ROOT]/alternative-registry`)` is already installed, use --force to override |
| [WARNING] be sure to add [..] |
| ") |
| .run(); |
| // "Updating" is not displayed here due to the --version fast-path. |
| cargo_process("install foo --registry alternative --version 1.0.0+abc") |
| .with_stderr("\ |
| [IGNORED] package `foo v1.0.0+abc (registry `[ROOT]/alternative-registry`)` is already installed, use --force to override |
| [WARNING] be sure to add [..] |
| ") |
| .run(); |
| cargo_process("install foo --registry alternative --version 1.0.0 --force") |
| .with_stderr( |
| "\ |
| [UPDATING] `[ROOT]/alternative-registry` index |
| [INSTALLING] foo v1.0.0+abc (registry `[ROOT]/alternative-registry`) |
| [COMPILING] foo v1.0.0+abc (registry `[ROOT]/alternative-registry`) |
| [FINISHED] [..] |
| [REPLACING] [ROOT]/home/.cargo/bin/foo[EXE] |
| [REPLACED] package [..] |
| [WARNING] be sure to add [..] |
| ", |
| ) |
| .run(); |
| // Check that from a fresh cache will work without metadata, too. |
| paths::home().join(".cargo/registry").rm_rf(); |
| paths::home().join(".cargo/bin").rm_rf(); |
| cargo_process("install foo --registry alternative --version 1.0.0") |
| .with_stderr("\ |
| [UPDATING] `[ROOT]/alternative-registry` index |
| [DOWNLOADING] crates ... |
| [DOWNLOADED] foo v1.0.0+abc (registry `[ROOT]/alternative-registry`) |
| [INSTALLING] foo v1.0.0+abc (registry `[ROOT]/alternative-registry`) |
| [COMPILING] foo v1.0.0+abc (registry `[ROOT]/alternative-registry`) |
| [FINISHED] [..] |
| [INSTALLING] [ROOT]/home/.cargo/bin/foo[EXE] |
| [INSTALLED] package `foo v1.0.0+abc (registry `[ROOT]/alternative-registry`)` (executable `foo[EXE]`) |
| [WARNING] be sure to add [..] |
| ") |
| .run(); |
| } |