From d323ad672cf9f75ff9f68c83ff963010cde61e48 Mon Sep 17 00:00:00 2001 From: Weihang Lo Date: Thu, 27 Jan 2022 17:50:43 +0800 Subject: [PATCH 1/3] Test: install with path outside current workspace Make up this test case to reflect the current incorrect behavior. --- tests/testsuite/install.rs | 53 +++++++++++++++++++++++++++++++++++++- 1 file changed, 52 insertions(+), 1 deletion(-) diff --git a/tests/testsuite/install.rs b/tests/testsuite/install.rs index c4a63edb39f..170f9088e1e 100644 --- a/tests/testsuite/install.rs +++ b/tests/testsuite/install.rs @@ -7,7 +7,7 @@ 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, + basic_manifest, cargo_process, no_such_file_err_msg, project, project_in, symlink_supported, t, }; use cargo_test_support::install::{ @@ -493,6 +493,57 @@ but found cargo.toml please try to rename it to Cargo.toml. --path must point to .run(); } +#[cargo_test] +fn install_relative_path_outside_current_ws() { + 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 = [] + edition = "2021" + + [dependencies] + foo = "1" + "#, + ) + .file("baz/src/lib.rs", "") + .build(); + + let _bin_project = project_in("bar") + .file("src/main.rs", "fn main() {}") + .build(); + + p.cargo("install --path ../bar/foo") + .with_status(101) + .with_stderr( + "\ +[ERROR] current package believes it's in a workspace when it's not: +current: [CWD]/../bar/foo/Cargo.toml +workspace: [CWD]/Cargo.toml + +this may be fixable by adding `../bar/foo` to the `workspace.members` [..] +Alternatively, [..] +", + ) + .run(); +} + #[cargo_test] fn multiple_crates_error() { let p = git::repo(&paths::root().join("foo")) From 44f650f2606c0cff8d8cce64b309dff4bda6ed99 Mon Sep 17 00:00:00 2001 From: Weihang Lo Date: Thu, 27 Jan 2022 17:58:06 +0800 Subject: [PATCH 2/3] Normalize `--path` to install bin outside current workspace For `Workspace::find_root`, `cargo_util::path::PathAncestors` won't do path normalization while walking back the ancestors. The responsibility lies in the caller. Thus, `cargo install` should normalize its `--path` argument before passing in `SourceId::for_path` and `Workspace::new`. `Config::reload_rooted_at` is not affected because cargo always starts searching and merging configs from where it is invoked. --- src/bin/cargo/commands/install.rs | 24 +++++++++++++++++++----- 1 file changed, 19 insertions(+), 5 deletions(-) diff --git a/src/bin/cargo/commands/install.rs b/src/bin/cargo/commands/install.rs index 621558d97c4..f0bbb4d7971 100644 --- a/src/bin/cargo/commands/install.rs +++ b/src/bin/cargo/commands/install.rs @@ -1,9 +1,11 @@ use crate::command_prelude::*; -use cargo::core::{GitReference, SourceId}; +use cargo::core::{GitReference, SourceId, Workspace}; use cargo::ops; use cargo::util::IntoUrl; +use cargo_util::paths; + pub fn cli() -> App { subcommand("install") .about("Install a Rust binary. Default location is $HOME/.cargo/bin") @@ -80,13 +82,20 @@ pub fn cli() -> App { } pub fn exec(config: &mut Config, args: &ArgMatches) -> CliResult { - if let Some(path) = args.value_of_path("path", config) { + let path = args.value_of_path("path", config); + if let Some(path) = &path { config.reload_rooted_at(path)?; } else { // TODO: Consider calling set_search_stop_path(home). config.reload_rooted_at(config.home().clone().into_path_unlocked())?; } + // In general, we try to avoid normalizing paths in Cargo, + // but in these particular cases we need it to fix rust-lang/cargo#10283. + // (Handle `SourceId::for_path` and `Workspace::new`, + // but not `Config::reload_rooted_at` which is always cwd) + let path = path.map(|p| paths::normalize_path(&p)); + let krates = args .values_of("crate") .unwrap_or_default() @@ -106,7 +115,7 @@ pub fn exec(config: &mut Config, args: &ArgMatches) -> CliResult { GitReference::DefaultBranch }; SourceId::for_git(&url, gitref)? - } else if let Some(path) = args.value_of_path("path", config) { + } else if let Some(path) = &path { SourceId::for_path(&path)? } else if krates.is_empty() { from_cwd = true; @@ -125,9 +134,14 @@ pub fn exec(config: &mut Config, args: &ArgMatches) -> CliResult { // We only provide workspace information for local crate installation from // one of the following sources: // - From current working directory (only work for edition 2015). - // - From a specific local file path. - let workspace = if from_cwd || args.is_present("path") { + // - From a specific local file path (from `--path` arg). + // + // This workspace information is for emitting helpful messages from + // `ArgMatchesExt::compile_options` and won't affect the actual compilation. + let workspace = if from_cwd { args.workspace(config).ok() + } else if let Some(path) = &path { + Workspace::new(&path.join("Cargo.toml"), config).ok() } else { None }; From 76301ebab9263393de0d8193680cf3f17f95057b Mon Sep 17 00:00:00 2001 From: Weihang Lo Date: Thu, 27 Jan 2022 18:09:28 +0800 Subject: [PATCH 3/3] Test: install bin `with --path` outside current workspace --- tests/testsuite/install.rs | 23 ++++++++++++++++++----- 1 file changed, 18 insertions(+), 5 deletions(-) diff --git a/tests/testsuite/install.rs b/tests/testsuite/install.rs index 170f9088e1e..5442c1884c9 100644 --- a/tests/testsuite/install.rs +++ b/tests/testsuite/install.rs @@ -530,15 +530,28 @@ fn install_relative_path_outside_current_ws() { .build(); p.cargo("install --path ../bar/foo") + .with_stderr(&format!( + "\ +[INSTALLING] foo v0.0.1 ([..]/bar/foo) +[COMPILING] foo v0.0.1 ([..]/bar/foo) +[FINISHED] release [..] +[INSTALLING] {home}/bin/foo[EXE] +[INSTALLED] package `foo v0.0.1 ([..]/bar/foo)` (executable `foo[EXE]`) +[WARNING] be sure to add [..] +", + home = cargo_home().display(), + )) + .run(); + + // Validate the workspace error message to display available targets. + p.cargo("install --path ../bar/foo --bin") .with_status(101) .with_stderr( "\ -[ERROR] current package believes it's in a workspace when it's not: -current: [CWD]/../bar/foo/Cargo.toml -workspace: [CWD]/Cargo.toml +[ERROR] \"--bin\" takes one argument. +Available binaries: + foo -this may be fixable by adding `../bar/foo` to the `workspace.members` [..] -Alternatively, [..] ", ) .run();