diff --git a/src/bin/cargo/commands/install.rs b/src/bin/cargo/commands/install.rs index 8aaaeb87b0e4..dabdc79c3539 100644 --- a/src/bin/cargo/commands/install.rs +++ b/src/bin/cargo/commands/install.rs @@ -69,6 +69,7 @@ pub fn cli() -> Command { ) .arg(opt("root", "Directory to install packages into").value_name("DIR")) .arg(flag("force", "Force overwriting existing crates or binaries").short('f')) + .arg_dry_run("Perform all checks without installing (unstable)") .arg(flag("no-track", "Do not save tracking information")) .arg(flag( "list", @@ -200,7 +201,9 @@ pub fn exec(gctx: &mut GlobalContext, args: &ArgMatches) -> CliResult { compile_opts.build_config.requested_profile = args.get_profile_name("release", ProfileChecking::Custom)?; - + if args.dry_run() { + gctx.cli_unstable().fail_if_stable_opt("--dry-run", 11123)?; + } if args.flag("list") { ops::install_list(root, gctx)?; } else { @@ -213,6 +216,7 @@ pub fn exec(gctx: &mut GlobalContext, args: &ArgMatches) -> CliResult { &compile_opts, args.flag("force"), args.flag("no-track"), + args.dry_run(), )?; } Ok(()) diff --git a/src/cargo/core/compiler/build_config.rs b/src/cargo/core/compiler/build_config.rs index 0f66d6dbbc66..4c804f27b681 100644 --- a/src/cargo/core/compiler/build_config.rs +++ b/src/cargo/core/compiler/build_config.rs @@ -31,6 +31,8 @@ pub struct BuildConfig { pub build_plan: bool, /// Output the unit graph to stdout instead of actually compiling. pub unit_graph: bool, + /// `true` to avoid really compiling. + pub dry_run: bool, /// An optional override of the rustc process for primary units pub primary_unit_rustc: Option, /// A thread used by `cargo fix` to receive messages on a socket regarding @@ -112,6 +114,7 @@ impl BuildConfig { force_rebuild: false, build_plan: false, unit_graph: false, + dry_run: false, primary_unit_rustc: None, rustfix_diagnostic_server: Rc::new(RefCell::new(None)), export_dir: None, diff --git a/src/cargo/core/compiler/build_runner/mod.rs b/src/cargo/core/compiler/build_runner/mod.rs index 3a6cd2a707bc..c261732382dd 100644 --- a/src/cargo/core/compiler/build_runner/mod.rs +++ b/src/cargo/core/compiler/build_runner/mod.rs @@ -126,6 +126,23 @@ impl<'a, 'gctx> BuildRunner<'a, 'gctx> { }) } + pub fn dry_run(mut self) -> CargoResult> { + let _lock = self + .bcx + .gctx + .acquire_package_cache_lock(CacheLockMode::Shared)?; + self.lto = super::lto::generate(self.bcx)?; + self.prepare_units()?; + self.prepare()?; + self.check_collisions()?; + + for unit in &self.bcx.roots { + self.collect_tests_and_executables(unit)?; + } + + Ok(self.compilation) + } + /// Starts compilation, waits for it to finish, and returns information /// about the result of compilation. /// diff --git a/src/cargo/ops/cargo_compile/mod.rs b/src/cargo/ops/cargo_compile/mod.rs index 77f6266355ff..17aaa922b8a2 100644 --- a/src/cargo/ops/cargo_compile/mod.rs +++ b/src/cargo/ops/cargo_compile/mod.rs @@ -156,7 +156,11 @@ pub fn compile_ws<'a>( } crate::core::gc::auto_gc(bcx.gctx); let build_runner = BuildRunner::new(&bcx)?; - build_runner.compile(exec) + if options.build_config.dry_run { + build_runner.dry_run() + } else { + build_runner.compile(exec) + } } /// Executes `rustc --print `. diff --git a/src/cargo/ops/cargo_install.rs b/src/cargo/ops/cargo_install.rs index 7abc1b393757..7b6b526ed9dc 100644 --- a/src/cargo/ops/cargo_install.rs +++ b/src/cargo/ops/cargo_install.rs @@ -297,7 +297,7 @@ impl<'gctx> InstallablePackage<'gctx> { Ok(duplicates) } - fn install_one(mut self) -> CargoResult { + fn install_one(mut self, dry_run: bool) -> CargoResult { self.gctx.shell().status("Installing", &self.pkg)?; let dst = self.root.join("bin").into_path_unlocked(); @@ -321,6 +321,7 @@ impl<'gctx> InstallablePackage<'gctx> { self.check_yanked_install()?; let exec: Arc = Arc::new(DefaultExecutor); + self.opts.build_config.dry_run = dry_run; let compile = ops::compile_ws(&self.ws, &self.opts, &exec).with_context(|| { if let Some(td) = td_opt.take() { // preserve the temporary directory, so the user can inspect it @@ -422,7 +423,7 @@ impl<'gctx> InstallablePackage<'gctx> { for &(bin, src) in binaries.iter() { let dst = staging_dir.path().join(bin); // Try to move if `target_dir` is transient. - if !self.source_id.is_path() && fs::rename(src, &dst).is_ok() { + if dry_run || (!self.source_id.is_path() && fs::rename(src, &dst).is_ok()) { continue; } paths::copy(src, &dst)?; @@ -441,11 +442,13 @@ impl<'gctx> InstallablePackage<'gctx> { let src = staging_dir.path().join(bin); let dst = dst.join(bin); self.gctx.shell().status("Installing", dst.display())?; - fs::rename(&src, &dst).with_context(|| { - format!("failed to move `{}` to `{}`", src.display(), dst.display()) - })?; - installed.bins.push(dst); - successful_bins.insert(bin.to_string()); + if !dry_run { + fs::rename(&src, &dst).with_context(|| { + format!("failed to move `{}` to `{}`", src.display(), dst.display()) + })?; + installed.bins.push(dst); + successful_bins.insert(bin.to_string()); + } } // Repeat for binaries which replace existing ones but don't pop the error @@ -456,10 +459,12 @@ impl<'gctx> InstallablePackage<'gctx> { let src = staging_dir.path().join(bin); let dst = dst.join(bin); self.gctx.shell().status("Replacing", dst.display())?; - fs::rename(&src, &dst).with_context(|| { - format!("failed to move `{}` to `{}`", src.display(), dst.display()) - })?; - successful_bins.insert(bin.to_string()); + if !dry_run { + fs::rename(&src, &dst).with_context(|| { + format!("failed to move `{}` to `{}`", src.display(), dst.display()) + })?; + successful_bins.insert(bin.to_string()); + } } Ok(()) }; @@ -476,9 +481,14 @@ impl<'gctx> InstallablePackage<'gctx> { &self.rustc.verbose_version, ); - if let Err(e) = - remove_orphaned_bins(&self.ws, &mut tracker, &duplicates, &self.pkg, &dst) - { + if let Err(e) = remove_orphaned_bins( + &self.ws, + &mut tracker, + &duplicates, + &self.pkg, + &dst, + dry_run, + ) { // Don't hard error on remove. self.gctx .shell() @@ -515,7 +525,10 @@ impl<'gctx> InstallablePackage<'gctx> { } } - if duplicates.is_empty() { + if dry_run { + self.gctx.shell().warn("aborting install due to dry run")?; + Ok(true) + } else if duplicates.is_empty() { self.gctx.shell().status( "Installed", format!( @@ -620,6 +633,7 @@ pub fn install( opts: &ops::CompileOptions, force: bool, no_track: bool, + dry_run: bool, ) -> CargoResult<()> { let root = resolve_root(root, gctx)?; let dst = root.join("bin").into_path_unlocked(); @@ -654,7 +668,7 @@ pub fn install( )?; let mut installed_anything = true; if let Some(installable_pkg) = installable_pkg { - installed_anything = installable_pkg.install_one()?; + installed_anything = installable_pkg.install_one(dry_run)?; } (installed_anything, false) } else { @@ -705,7 +719,7 @@ pub fn install( let install_results: Vec<_> = pkgs_to_install .into_iter() - .map(|(krate, installable_pkg)| (krate, installable_pkg.install_one())) + .map(|(krate, installable_pkg)| (krate, installable_pkg.install_one(dry_run))) .collect(); for (krate, result) in install_results { @@ -857,6 +871,7 @@ fn remove_orphaned_bins( duplicates: &BTreeMap>, pkg: &Package, dst: &Path, + dry_run: bool, ) -> CargoResult<()> { let filter = ops::CompileFilter::new_all_targets(); let all_self_names = exe_names(pkg, &filter); @@ -894,8 +909,10 @@ fn remove_orphaned_bins( old_pkg ), )?; - paths::remove_file(&full_path) - .with_context(|| format!("failed to remove {:?}", full_path))?; + if !dry_run { + paths::remove_file(&full_path) + .with_context(|| format!("failed to remove {:?}", full_path))?; + } } } } diff --git a/src/doc/man/cargo-install.md b/src/doc/man/cargo-install.md index fa593206612e..114284f5d4e6 100644 --- a/src/doc/man/cargo-install.md +++ b/src/doc/man/cargo-install.md @@ -121,6 +121,10 @@ Filesystem path to local crate to install from. List all installed packages and their versions. {{/option}} +{{#option "`-n`" "`--dry-run`" }} +(unstable) Perform all checks without installing. +{{/option}} + {{#option "`-f`" "`--force`" }} Force overwriting existing crates or binaries. This can be used if a package has installed a binary with the same name as another package. This is also diff --git a/src/doc/man/generated_txt/cargo-install.txt b/src/doc/man/generated_txt/cargo-install.txt index e5d1182ff055..a9deef68143a 100644 --- a/src/doc/man/generated_txt/cargo-install.txt +++ b/src/doc/man/generated_txt/cargo-install.txt @@ -120,6 +120,9 @@ OPTIONS --list List all installed packages and their versions. + -n, --dry-run + (unstable) Perform all checks without installing. + -f, --force Force overwriting existing crates or binaries. This can be used if a package has installed a binary with the same name as another diff --git a/src/doc/src/commands/cargo-install.md b/src/doc/src/commands/cargo-install.md index 78efb92e8c04..64693b50c7d5 100644 --- a/src/doc/src/commands/cargo-install.md +++ b/src/doc/src/commands/cargo-install.md @@ -124,6 +124,11 @@ treated as a caret requirement like Cargo dependencies are.
List all installed packages and their versions.
+
-n
+
--dry-run
+
(unstable) Perform all checks without installing.
+ +
-f
--force
Force overwriting existing crates or binaries. This can be used if a package diff --git a/src/etc/man/cargo-install.1 b/src/etc/man/cargo-install.1 index 8f524bc7cea8..8bf4ded8bc17 100644 --- a/src/etc/man/cargo-install.1 +++ b/src/etc/man/cargo-install.1 @@ -151,6 +151,12 @@ Filesystem path to local crate to install from. List all installed packages and their versions. .RE .sp +\fB\-n\fR, +\fB\-\-dry\-run\fR +.RS 4 +(unstable) Perform all checks without installing. +.RE +.sp \fB\-f\fR, \fB\-\-force\fR .RS 4 diff --git a/tests/testsuite/cargo_install/help/stdout.term.svg b/tests/testsuite/cargo_install/help/stdout.term.svg index e021922563f3..38ef23809e74 100644 --- a/tests/testsuite/cargo_install/help/stdout.term.svg +++ b/tests/testsuite/cargo_install/help/stdout.term.svg @@ -1,4 +1,4 @@ - +