From 72e6b9d32619d274511d0749cb1149f32a69decf Mon Sep 17 00:00:00 2001 From: Mike Shal Date: Thu, 3 Aug 2017 10:02:00 -0400 Subject: [PATCH 1/2] Add --build-plan for 'cargo build' With 'cargo build --build-plan', cargo does not actually run any commands, but instead prints out what it would have done in the form of a JSON data structure. Fixes #3815 --- src/bin/cargo/command_prelude.rs | 10 + src/bin/cargo/commands/build.rs | 1 + src/cargo/core/compiler/build_config.rs | 3 + src/cargo/core/compiler/build_context/mod.rs | 12 ++ src/cargo/core/compiler/build_plan.rs | 158 ++++++++++++++ src/cargo/core/compiler/context/mod.rs | 41 +++- src/cargo/core/compiler/custom_build.rs | 113 +++++----- src/cargo/core/compiler/job_queue.rs | 37 +++- src/cargo/core/compiler/mod.rs | 39 +++- tests/testsuite/build_plan.rs | 207 +++++++++++++++++++ tests/testsuite/main.rs | 1 + 11 files changed, 547 insertions(+), 75 deletions(-) create mode 100644 src/cargo/core/compiler/build_plan.rs create mode 100644 tests/testsuite/build_plan.rs diff --git a/src/bin/cargo/command_prelude.rs b/src/bin/cargo/command_prelude.rs index 3ef518e57ef..6db646be828 100644 --- a/src/bin/cargo/command_prelude.rs +++ b/src/bin/cargo/command_prelude.rs @@ -132,6 +132,10 @@ pub trait AppExt: Sized { ) } + fn arg_build_plan(self) -> Self { + self._arg(opt("build-plan", "Output the build plan in JSON")) + } + fn arg_new_opts(self) -> Self { self._arg( opt( @@ -275,6 +279,12 @@ pub trait ArgMatchesExt { let mut build_config = BuildConfig::new(config, self.jobs()?, &self.target(), mode)?; build_config.message_format = message_format; build_config.release = self._is_present("release"); + build_config.build_plan = self._is_present("build-plan"); + if build_config.build_plan && !config.cli_unstable().unstable_options { + Err(format_err!( + "`--build-plan` flag is unstable, pass `-Z unstable-options` to enable it" + ))?; + }; let opts = CompileOptions { config, diff --git a/src/bin/cargo/commands/build.rs b/src/bin/cargo/commands/build.rs index 5dbc3850a4e..f1c8b256b2e 100644 --- a/src/bin/cargo/commands/build.rs +++ b/src/bin/cargo/commands/build.rs @@ -31,6 +31,7 @@ pub fn cli() -> App { .arg(opt("out-dir", "Copy final artifacts to this directory").value_name("PATH")) .arg_manifest_path() .arg_message_format() + .arg_build_plan() .after_help( "\ If the --package argument is given, then SPEC is a package id specification diff --git a/src/cargo/core/compiler/build_config.rs b/src/cargo/core/compiler/build_config.rs index e4d1377b78f..7eb863f05e4 100644 --- a/src/cargo/core/compiler/build_config.rs +++ b/src/cargo/core/compiler/build_config.rs @@ -14,6 +14,8 @@ pub struct BuildConfig { pub mode: CompileMode, /// Whether to print std output in json format (for machine reading) pub message_format: MessageFormat, + /// Output a build plan to stdout instead of actually compiling. + pub build_plan: bool, } impl BuildConfig { @@ -87,6 +89,7 @@ impl BuildConfig { release: false, mode, message_format: MessageFormat::Human, + build_plan: false, }) } diff --git a/src/cargo/core/compiler/build_context/mod.rs b/src/cargo/core/compiler/build_context/mod.rs index 1e7112b9abe..c49d6c5bd7f 100644 --- a/src/cargo/core/compiler/build_context/mod.rs +++ b/src/cargo/core/compiler/build_context/mod.rs @@ -229,6 +229,18 @@ impl<'a, 'cfg> BuildContext<'a, 'cfg> { } None } + + /// Return the list of filenames read by cargo to generate the BuildContext + /// (all Cargo.toml, etc). + pub fn inputs(&self) -> CargoResult> { + let mut inputs = Vec::new(); + for id in self.packages.package_ids() { + let pkg = self.get_package(id)?; + inputs.push(pkg.manifest_path().to_path_buf()); + } + inputs.sort(); + Ok(inputs) + } } /// Information required to build for a target diff --git a/src/cargo/core/compiler/build_plan.rs b/src/cargo/core/compiler/build_plan.rs new file mode 100644 index 00000000000..788d569b8fa --- /dev/null +++ b/src/cargo/core/compiler/build_plan.rs @@ -0,0 +1,158 @@ +//! A graph-like structure used to represent the rustc commands to build the project and the +//! interdependencies between them. +//! +//! The BuildPlan structure is used to store the dependency graph of a dry run so that it can be +//! shared with an external build system. Each Invocation in the BuildPlan comprises a single +//! subprocess and defines the build environment, the outputs produced by the subprocess, and the +//! dependencies on other Invocations. + +use std::collections::BTreeMap; + +use core::TargetKind; +use super::{Context, Kind, Unit}; +use super::context::OutputFile; +use util::{internal, CargoResult, ProcessBuilder}; +use std::sync::Arc; +use std::path::PathBuf; +use serde_json; +use semver; + +#[derive(Debug, Serialize)] +struct Invocation { + package_name: String, + package_version: semver::Version, + target_kind: TargetKind, + kind: Kind, + deps: Vec, + outputs: Vec, + links: BTreeMap, + program: String, + args: Vec, + env: BTreeMap, + cwd: Option, +} + +#[derive(Debug)] +pub struct BuildPlan { + invocation_map: BTreeMap, + plan: SerializedBuildPlan, +} + +#[derive(Debug, Serialize)] +struct SerializedBuildPlan { + invocations: Vec, + inputs: Vec, +} + +impl Invocation { + pub fn new(unit: &Unit, deps: Vec) -> Invocation { + let id = unit.pkg.package_id(); + Invocation { + package_name: id.name().to_string(), + package_version: id.version().clone(), + kind: unit.kind, + target_kind: unit.target.kind().clone(), + deps: deps, + outputs: Vec::new(), + links: BTreeMap::new(), + program: String::new(), + args: Vec::new(), + env: BTreeMap::new(), + cwd: None, + } + } + + pub fn add_output(&mut self, path: &PathBuf, link: &Option) { + self.outputs.push(path.clone()); + if let Some(ref link) = *link { + self.links.insert(link.clone(), path.clone()); + } + } + + pub fn update_cmd(&mut self, cmd: ProcessBuilder) -> CargoResult<()> { + self.program = cmd.get_program() + .to_str() + .ok_or_else(|| format_err!("unicode program string required"))? + .to_string() + .clone(); + self.cwd = Some(cmd.get_cwd().unwrap().to_path_buf()); + for arg in cmd.get_args().iter() { + self.args.push( + arg.to_str() + .ok_or_else(|| format_err!("unicode argument string required"))? + .to_string() + .clone(), + ); + } + for var in cmd.get_envs().keys() { + let value = cmd.get_env(var).unwrap_or_default(); + self.env.insert( + var.clone(), + value + .to_str() + .ok_or_else(|| format_err!("unicode environment value required"))? + .to_string(), + ); + } + Ok(()) + } +} + +impl BuildPlan { + pub fn new() -> BuildPlan { + BuildPlan { + invocation_map: BTreeMap::new(), + plan: SerializedBuildPlan::new(), + } + } + + pub fn add(&mut self, cx: &Context, unit: &Unit) -> CargoResult<()> { + let id = self.plan.invocations.len(); + self.invocation_map.insert(unit.buildkey(), id); + let deps = cx.dep_targets(&unit) + .iter() + .map(|dep| self.invocation_map[&dep.buildkey()]) + .collect(); + let invocation = Invocation::new(unit, deps); + self.plan.invocations.push(invocation); + Ok(()) + } + + pub fn update( + &mut self, + invocation_name: String, + cmd: ProcessBuilder, + outputs: Arc>, + ) -> CargoResult<()> { + let id = self.invocation_map[&invocation_name]; + let invocation = self.plan + .invocations + .get_mut(id) + .ok_or_else(|| internal(format!("couldn't find invocation for {}", invocation_name)))?; + + invocation.update_cmd(cmd)?; + for output in outputs.iter() { + invocation.add_output(&output.path, &output.hardlink); + } + + Ok(()) + } + + pub fn set_inputs(&mut self, inputs: Vec) { + self.plan.inputs = inputs; + } + + pub fn output_plan(self) { + let encoded = serde_json::to_string(&self.plan).unwrap(); + println!("{}", encoded); + } +} + +impl SerializedBuildPlan { + pub fn new() -> SerializedBuildPlan { + SerializedBuildPlan { + invocations: Vec::new(), + inputs: Vec::new(), + } + } +} diff --git a/src/cargo/core/compiler/context/mod.rs b/src/cargo/core/compiler/context/mod.rs index af04750a858..0000b895057 100644 --- a/src/cargo/core/compiler/context/mod.rs +++ b/src/cargo/core/compiler/context/mod.rs @@ -3,26 +3,28 @@ use std::collections::{HashMap, HashSet}; use std::fmt::Write; use std::path::PathBuf; use std::sync::Arc; +use std::cmp::Ordering; use jobserver::Client; use core::{Package, PackageId, Resolve, Target}; use core::profiles::Profile; use util::errors::{CargoResult, CargoResultExt}; -use util::{internal, profile, Config}; +use util::{internal, profile, Config, short_hash}; use super::custom_build::{self, BuildDeps, BuildScripts, BuildState}; use super::fingerprint::Fingerprint; use super::job_queue::JobQueue; use super::layout::Layout; use super::{BuildContext, Compilation, CompileMode, Executor, FileFlavor, Kind}; +use super::build_plan::BuildPlan; mod unit_dependencies; use self::unit_dependencies::build_unit_dependencies; mod compilation_files; -pub use self::compilation_files::Metadata; -use self::compilation_files::{CompilationFiles, OutputFile}; +pub use self::compilation_files::{Metadata, OutputFile}; +use self::compilation_files::CompilationFiles; /// All information needed to define a Unit. /// @@ -62,6 +64,24 @@ pub struct Unit<'a> { pub mode: CompileMode, } +impl<'a> Unit<'a> { + pub fn buildkey(&self) -> String { + format!("{}-{}", self.pkg.name(), short_hash(self)) + } +} + +impl<'a> Ord for Unit<'a> { + fn cmp(&self, other: &Unit) -> Ordering { + self.buildkey().cmp(&other.buildkey()) + } +} + +impl<'a> PartialOrd for Unit<'a> { + fn partial_cmp(&self, other: &Unit) -> Option { + Some(self.cmp(other)) + } +} + pub struct Context<'a, 'cfg: 'a> { pub bcx: &'a BuildContext<'a, 'cfg>, pub compilation: Compilation<'cfg>, @@ -121,6 +141,8 @@ impl<'a, 'cfg> Context<'a, 'cfg> { exec: &Arc, ) -> CargoResult> { let mut queue = JobQueue::new(self.bcx); + let mut plan = BuildPlan::new(); + let build_plan = self.bcx.build_config.build_plan; self.prepare_units(export_dir, units)?; self.prepare()?; custom_build::build_map(&mut self, units)?; @@ -131,11 +153,16 @@ impl<'a, 'cfg> Context<'a, 'cfg> { // part of this, that's all done next as part of the `execute` // function which will run everything in order with proper // parallelism. - super::compile(&mut self, &mut queue, unit, exec)?; + super::compile(&mut self, &mut queue, &mut plan, unit, exec)?; } // Now that we've figured out everything that we're going to do, do it! - queue.execute(&mut self)?; + queue.execute(&mut self, &mut plan)?; + + if build_plan { + plan.set_inputs(self.bcx.inputs()?); + plan.output_plan(); + } for unit in units.iter() { for output in self.outputs(unit)?.iter() { @@ -366,7 +393,9 @@ impl<'a, 'cfg> Context<'a, 'cfg> { return Vec::new(); } } - self.unit_dependencies[unit].clone() + let mut deps = self.unit_dependencies[unit].clone(); + deps.sort(); + deps } pub fn incremental_args(&self, unit: &Unit) -> CargoResult> { diff --git a/src/cargo/core/compiler/custom_build.rs b/src/cargo/core/compiler/custom_build.rs index f358a5eaa4f..48588bb25ac 100644 --- a/src/cargo/core/compiler/custom_build.rs +++ b/src/cargo/core/compiler/custom_build.rs @@ -94,11 +94,15 @@ pub fn prepare<'a, 'cfg>( build_work(cx, unit)? }; - // Now that we've prep'd our work, build the work needed to manage the - // fingerprint and then start returning that upwards. - let (freshness, dirty, fresh) = fingerprint::prepare_build_cmd(cx, unit)?; + if cx.bcx.build_config.build_plan { + Ok((work_dirty, work_fresh, Freshness::Dirty)) + } else { + // Now that we've prep'd our work, build the work needed to manage the + // fingerprint and then start returning that upwards. + let (freshness, dirty, fresh) = fingerprint::prepare_build_cmd(cx, unit)?; - Ok((work_dirty.then(dirty), work_fresh.then(fresh), freshness)) + Ok((work_dirty.then(dirty), work_fresh.then(fresh), freshness)) + } } fn build_work<'a, 'cfg>(cx: &mut Context<'a, 'cfg>, unit: &Unit<'a>) -> CargoResult<(Work, Work)> { @@ -111,6 +115,8 @@ fn build_work<'a, 'cfg>(cx: &mut Context<'a, 'cfg>, unit: &Unit<'a>) -> CargoRes .expect("running a script not depending on an actual script"); let script_output = cx.files().build_script_dir(build_script_unit); let build_output = cx.files().build_script_out_dir(unit); + let build_plan = bcx.build_config.build_plan; + let invocation_name = unit.buildkey(); // Building the command to execute let to_exec = script_output.join(unit.target.name()); @@ -269,7 +275,7 @@ fn build_work<'a, 'cfg>(cx: &mut Context<'a, 'cfg>, unit: &Unit<'a>) -> CargoRes // along to this custom build command. We're also careful to augment our // dynamic library search path in case the build script depended on any // native dynamic libraries. - { + if !build_plan { let build_state = build_state.outputs.lock().unwrap(); for (name, id) in lib_deps { let key = (id.clone(), kind); @@ -294,54 +300,57 @@ fn build_work<'a, 'cfg>(cx: &mut Context<'a, 'cfg>, unit: &Unit<'a>) -> CargoRes } // And now finally, run the build command itself! - state.running(&cmd); - let output = cmd.exec_with_streaming( - &mut |out_line| { - state.stdout(out_line); - Ok(()) - }, - &mut |err_line| { - state.stderr(err_line); - Ok(()) - }, - true, - ).map_err(|e| { - format_err!( - "failed to run custom build command for `{}`\n{}", - pkg_name, - e - ) - })?; - - // After the build command has finished running, we need to be sure to - // remember all of its output so we can later discover precisely what it - // was, even if we don't run the build command again (due to freshness). - // - // This is also the location where we provide feedback into the build - // state informing what variables were discovered via our script as - // well. - paths::write(&output_file, &output.stdout)?; - paths::write(&err_file, &output.stderr)?; - paths::write(&root_output_file, util::path2bytes(&root_output)?)?; - let parsed_output = - BuildOutput::parse(&output.stdout, &pkg_name, &root_output, &root_output)?; - - if json_messages { - let library_paths = parsed_output - .library_paths - .iter() - .map(|l| l.display().to_string()) - .collect::>(); - machine_message::emit(&machine_message::BuildScript { - package_id: &id, - linked_libs: &parsed_output.library_links, - linked_paths: &library_paths, - cfgs: &parsed_output.cfgs, - env: &parsed_output.env, - }); - } + if build_plan { + state.build_plan(invocation_name, cmd.clone(), Arc::new(Vec::new())); + } else { + state.running(&cmd); + let output = cmd.exec_with_streaming( + &mut |out_line| { + state.stdout(out_line); + Ok(()) + }, + &mut |err_line| { + state.stderr(err_line); + Ok(()) + }, + true, + ).map_err(|e| { + format_err!( + "failed to run custom build command for `{}`\n{}", + pkg_name, + e + ) + })?; - build_state.insert(id, kind, parsed_output); + // After the build command has finished running, we need to be sure to + // remember all of its output so we can later discover precisely what it + // was, even if we don't run the build command again (due to freshness). + // + // This is also the location where we provide feedback into the build + // state informing what variables were discovered via our script as + // well. + paths::write(&output_file, &output.stdout)?; + paths::write(&err_file, &output.stderr)?; + paths::write(&root_output_file, util::path2bytes(&root_output)?)?; + let parsed_output = + BuildOutput::parse(&output.stdout, &pkg_name, &root_output, &root_output)?; + + if json_messages { + let library_paths = parsed_output + .library_paths + .iter() + .map(|l| l.display().to_string()) + .collect::>(); + machine_message::emit(&machine_message::BuildScript { + package_id: &id, + linked_libs: &parsed_output.library_links, + linked_paths: &library_paths, + cfgs: &parsed_output.cfgs, + env: &parsed_output.env, + }); + } + build_state.insert(id, kind, parsed_output); + } Ok(()) }); diff --git a/src/cargo/core/compiler/job_queue.rs b/src/cargo/core/compiler/job_queue.rs index 235dccc93e6..a7c39f9c544 100644 --- a/src/cargo/core/compiler/job_queue.rs +++ b/src/cargo/core/compiler/job_queue.rs @@ -4,6 +4,7 @@ use std::fmt; use std::io; use std::mem; use std::sync::mpsc::{channel, Receiver, Sender}; +use std::sync::Arc; use crossbeam::{self, Scope}; use jobserver::{Acquired, HelperThread}; @@ -15,7 +16,8 @@ use util::{internal, profile, CargoResult, CargoResultExt, ProcessBuilder}; use util::{Config, DependencyQueue, Dirty, Fresh, Freshness}; use super::job::Job; -use super::{BuildContext, CompileMode, Context, Kind, Unit}; +use super::{BuildContext, BuildPlan, CompileMode, Context, Kind, Unit}; +use super::context::OutputFile; /// A management structure of the entire dependency graph to compile. /// @@ -58,6 +60,7 @@ pub struct JobState<'a> { enum Message<'a> { Run(String), + BuildPlanMsg(String, ProcessBuilder, Arc>), Stdout(String), Stderr(String), Token(io::Result), @@ -69,6 +72,16 @@ impl<'a> JobState<'a> { let _ = self.tx.send(Message::Run(cmd.to_string())); } + pub fn build_plan( + &self, + module_name: String, + cmd: ProcessBuilder, + filenames: Arc>, + ) { + let _ = self.tx + .send(Message::BuildPlanMsg(module_name, cmd, filenames)); + } + pub fn stdout(&self, out: &str) { let _ = self.tx.send(Message::Stdout(out.to_string())); } @@ -115,7 +128,7 @@ impl<'a> JobQueue<'a> { /// This function will spawn off `config.jobs()` workers to build all of the /// necessary dependencies, in order. Freshness is propagated as far as /// possible along each dependency chain. - pub fn execute(&mut self, cx: &mut Context) -> CargoResult<()> { + pub fn execute(&mut self, cx: &mut Context, plan: &mut BuildPlan) -> CargoResult<()> { let _p = profile::start("executing the job graph"); self.queue.queue_finished(); @@ -141,17 +154,19 @@ impl<'a> JobQueue<'a> { }) .chain_err(|| "failed to create helper thread for jobserver management")?; - crossbeam::scope(|scope| self.drain_the_queue(cx, scope, &helper)) + crossbeam::scope(|scope| self.drain_the_queue(cx, plan, scope, &helper)) } fn drain_the_queue( &mut self, cx: &mut Context, + plan: &mut BuildPlan, scope: &Scope<'a>, jobserver_helper: &HelperThread, ) -> CargoResult<()> { let mut tokens = Vec::new(); let mut queue = Vec::new(); + let build_plan = cx.bcx.build_config.build_plan; trace!("queue: {:#?}", self.queue); // Iteratively execute the entire dependency graph. Each turn of the @@ -192,7 +207,7 @@ impl<'a> JobQueue<'a> { // we're able to perform some parallel work. while error.is_none() && self.active < tokens.len() + 1 && !queue.is_empty() { let (key, job, fresh) = queue.remove(0); - self.run(key, fresh, job, cx.bcx.config, scope)?; + self.run(key, fresh, job, cx.bcx.config, scope, build_plan)?; } // If after all that we're not actually running anything then we're @@ -215,6 +230,9 @@ impl<'a> JobQueue<'a> { .shell() .verbose(|c| c.status("Running", &cmd))?; } + Message::BuildPlanMsg(module_name, cmd, filenames) => { + plan.update(module_name, cmd, filenames)?; + } Message::Stdout(out) => { if cx.bcx.config.extra_verbose() { println!("{}", out); @@ -303,7 +321,9 @@ impl<'a> JobQueue<'a> { "{} [{}] target(s) in {}", build_type, opt_type, time_elapsed ); - cx.bcx.config.shell().status("Finished", message)?; + if !build_plan { + cx.bcx.config.shell().status("Finished", message)?; + } Ok(()) } else if let Some(e) = error { Err(e) @@ -322,6 +342,7 @@ impl<'a> JobQueue<'a> { job: Job, config: &Config, scope: &Scope<'a>, + build_plan: bool, ) -> CargoResult<()> { info!("start: {:?}", key); @@ -340,8 +361,10 @@ impl<'a> JobQueue<'a> { } } - // Print out some nice progress information - self.note_working_on(config, &key, fresh)?; + if !build_plan { + // Print out some nice progress information + self.note_working_on(config, &key, fresh)?; + } Ok(()) } diff --git a/src/cargo/core/compiler/mod.rs b/src/cargo/core/compiler/mod.rs index 8b503aaae6a..e3f943661cf 100644 --- a/src/cargo/core/compiler/mod.rs +++ b/src/cargo/core/compiler/mod.rs @@ -16,6 +16,7 @@ use util::paths; use util::{self, machine_message, Freshness, ProcessBuilder}; use util::{internal, join_paths, profile}; +use self::build_plan::BuildPlan; use self::job::{Job, Work}; use self::job_queue::JobQueue; @@ -30,6 +31,7 @@ pub use self::layout::is_bad_artifact_name; mod build_config; mod build_context; +mod build_plan; mod compilation; mod context; mod custom_build; @@ -42,7 +44,7 @@ mod output_depinfo; /// Whether an object is for the host arch, or the target arch. /// /// These will be the same unless cross-compiling. -#[derive(PartialEq, Eq, Hash, Debug, Clone, Copy, PartialOrd, Ord)] +#[derive(PartialEq, Eq, Hash, Debug, Clone, Copy, PartialOrd, Ord, Serialize)] pub enum Kind { Host, Target, @@ -93,10 +95,12 @@ impl Executor for DefaultExecutor {} fn compile<'a, 'cfg: 'a>( cx: &mut Context<'a, 'cfg>, jobs: &mut JobQueue<'a>, + plan: &mut BuildPlan, unit: &Unit<'a>, exec: &Arc, ) -> CargoResult<()> { let bcx = cx.bcx; + let build_plan = bcx.build_config.build_plan; if !cx.compiled.insert(*unit) { return Ok(()); } @@ -112,6 +116,12 @@ fn compile<'a, 'cfg: 'a>( } else if unit.mode == CompileMode::Doctest { // we run these targets later, so this is just a noop for now (Work::noop(), Work::noop(), Freshness::Fresh) + } else if build_plan { + ( + rustc(cx, unit, &exec.clone())?, + Work::noop(), + Freshness::Dirty, + ) } else { let (mut freshness, dirty, fresh) = fingerprint::prepare_target(cx, unit)?; let work = if unit.mode.is_doc() { @@ -134,7 +144,10 @@ fn compile<'a, 'cfg: 'a>( // Be sure to compile all dependencies of this target as well. for unit in cx.dep_targets(unit).iter() { - compile(cx, jobs, unit, exec)?; + compile(cx, jobs, plan, unit, exec)?; + } + if build_plan { + plan.add(cx, unit)?; } Ok(()) @@ -146,8 +159,10 @@ fn rustc<'a, 'cfg>( exec: &Arc, ) -> CargoResult { let mut rustc = prepare_rustc(cx, &unit.target.rustc_crate_types(), unit)?; + let build_plan = cx.bcx.build_config.build_plan; let name = unit.pkg.name().to_string(); + let buildkey = unit.buildkey(); // If this is an upstream dep we don't want warnings from, turn off all // lints. @@ -209,14 +224,16 @@ fn rustc<'a, 'cfg>( // previous build scripts, we include them in the rustc invocation. if let Some(build_deps) = build_deps { let build_state = build_state.outputs.lock().unwrap(); - add_native_deps( - &mut rustc, - &build_state, - &build_deps, - pass_l_flag, - ¤t_id, - )?; - add_plugin_deps(&mut rustc, &build_state, &build_deps, &root_output)?; + if !build_plan { + add_native_deps( + &mut rustc, + &build_state, + &build_deps, + pass_l_flag, + ¤t_id, + )?; + add_plugin_deps(&mut rustc, &build_state, &build_deps, &root_output)?; + } add_custom_env(&mut rustc, &build_state, ¤t_id, kind)?; } @@ -268,6 +285,8 @@ fn rustc<'a, 'cfg>( Ok(()) }, ).chain_err(|| format!("Could not compile `{}`.", name))?; + } else if build_plan { + state.build_plan(buildkey, rustc.clone(), outputs.clone()); } else { exec.exec(rustc, &package_id, &target) .map_err(Internal::new) diff --git a/tests/testsuite/build_plan.rs b/tests/testsuite/build_plan.rs new file mode 100644 index 00000000000..d11eef1225b --- /dev/null +++ b/tests/testsuite/build_plan.rs @@ -0,0 +1,207 @@ +use cargotest::ChannelChanger; +use cargotest::support::{basic_bin_manifest, execs, main_file, project}; +use hamcrest::{assert_that, existing_file, is_not}; + +#[test] +fn cargo_build_plan_simple() { + let p = project("foo") + .file("Cargo.toml", &basic_bin_manifest("foo")) + .file("src/foo.rs", &main_file(r#""i am foo""#, &[])) + .build(); + + assert_that( + p.cargo("build") + .masquerade_as_nightly_cargo() + .arg("--build-plan") + .arg("-Zunstable-options"), + execs().with_status(0).with_json( + r#" + { + "inputs": [ + "[..][/]foo[/]Cargo.toml" + ], + "invocations": [ + { + "args": "{...}", + "cwd": "[..][/]target[/]cit[/][..][/]foo", + "deps": [], + "env": "{...}", + "kind": "Host", + "links": "{...}", + "outputs": "{...}", + "package_name": "foo", + "package_version": "0.5.0", + "program": "rustc", + "target_kind": ["bin"] + } + ] + } + "#, + ), + ); + assert_that(&p.bin("foo"), is_not(existing_file())); +} + +#[test] +fn cargo_build_plan_single_dep() { + let p = project("foo") + .file( + "Cargo.toml", + r#" + [package] + name = "foo" + authors = [] + version = "0.5.0" + + [dependencies] + bar = { path = "bar" } + "#, + ) + .file( + "src/lib.rs", + r#" + extern crate bar; + pub fn foo() { bar::bar(); } + + #[test] + fn test() { foo(); } + "#, + ) + .file( + "bar/Cargo.toml", + r#" + [package] + name = "bar" + version = "0.0.1" + authors = [] + "#, + ) + .file("bar/src/lib.rs", "pub fn bar() {}") + .build(); + assert_that( + p.cargo("build") + .masquerade_as_nightly_cargo() + .arg("--build-plan") + .arg("-Zunstable-options"), + execs().with_status(0).with_json( + r#" + { + "inputs": [ + "[..][/]foo[/]Cargo.toml", + "[..][/]foo[/]bar[/]Cargo.toml" + ], + "invocations": [ + { + "args": "{...}", + "cwd": "[..][/]target[/]cit[/][..][/]foo", + "deps": [], + "env": "{...}", + "kind": "Host", + "links": "{...}", + "outputs": [ + "[..][/]foo[/]target[/]debug[/]deps[/]libbar-[..].rlib" + ], + "package_name": "bar", + "package_version": "0.0.1", + "program": "rustc", + "target_kind": ["lib"] + }, + { + "args": "{...}", + "cwd": "[..][/]target[/]cit[/][..][/]foo", + "deps": [0], + "env": "{...}", + "kind": "Host", + "links": "{...}", + "outputs": [ + "[..][/]foo[/]target[/]debug[/]deps[/]libfoo-[..].rlib" + ], + "package_name": "foo", + "package_version": "0.5.0", + "program": "rustc", + "target_kind": ["lib"] + } + ] + } + "#, + ), + ); +} + +#[test] +fn cargo_build_plan_build_script() { + let p = project("foo") + .file( + "Cargo.toml", + r#" + [project] + + name = "foo" + version = "0.5.0" + authors = ["wycats@example.com"] + build = "build.rs" + "#, + ) + .file("src/main.rs", r#"fn main() {}"#) + .file("build.rs", r#"fn main() {}"#) + .build(); + + assert_that( + p.cargo("build") + .masquerade_as_nightly_cargo() + .arg("--build-plan") + .arg("-Zunstable-options"), + execs().with_status(0).with_json( + r#" + { + "inputs": [ + "[..][/]foo[/]Cargo.toml" + ], + "invocations": [ + { + "args": "{...}", + "cwd": "[..][/]target[/]cit[/][..][/]foo", + "deps": [], + "env": "{...}", + "kind": "Host", + "links": "{...}", + "outputs": [ + "[..][/]foo[/]target[/]debug[/]build[/][..][/]build_script_build-[..]" + ], + "package_name": "foo", + "package_version": "0.5.0", + "program": "rustc", + "target_kind": ["custom-build"] + }, + { + "args": "{...}", + "cwd": "[..][/]target[/]cit[/][..][/]foo", + "deps": [0], + "env": "{...}", + "kind": "Host", + "links": "{...}", + "outputs": [], + "package_name": "foo", + "package_version": "0.5.0", + "program": "[..][/]build-script-build", + "target_kind": ["custom-build"] + }, + { + "args": "{...}", + "cwd": "[..][/]target[/]cit[/][..][/]foo", + "deps": [1], + "env": "{...}", + "kind": "Host", + "links": "{...}", + "outputs": "{...}", + "package_name": "foo", + "package_version": "0.5.0", + "program": "rustc", + "target_kind": ["bin"] + } + ] + } + "#, + ), + ); +} diff --git a/tests/testsuite/main.rs b/tests/testsuite/main.rs index 413f0463d3f..b1b7496abb5 100644 --- a/tests/testsuite/main.rs +++ b/tests/testsuite/main.rs @@ -30,6 +30,7 @@ mod bench; mod build_auth; mod build_lib; mod build; +mod build_plan; mod build_script_env; mod build_script; mod cargo_alias_config; From a071c3ab9bf3d18f5c5b7a0c96fe670987680f9b Mon Sep 17 00:00:00 2001 From: Alex Crichton Date: Thu, 10 May 2018 12:07:02 -0700 Subject: [PATCH 2/2] Fixup some minor review comments --- src/cargo/core/compiler/build_plan.rs | 13 +++++++------ 1 file changed, 7 insertions(+), 6 deletions(-) diff --git a/src/cargo/core/compiler/build_plan.rs b/src/cargo/core/compiler/build_plan.rs index 788d569b8fa..d0ac68eede2 100644 --- a/src/cargo/core/compiler/build_plan.rs +++ b/src/cargo/core/compiler/build_plan.rs @@ -73,19 +73,20 @@ impl Invocation { self.program = cmd.get_program() .to_str() .ok_or_else(|| format_err!("unicode program string required"))? - .to_string() - .clone(); + .to_string(); self.cwd = Some(cmd.get_cwd().unwrap().to_path_buf()); for arg in cmd.get_args().iter() { self.args.push( arg.to_str() .ok_or_else(|| format_err!("unicode argument string required"))? - .to_string() - .clone(), + .to_string(), ); } - for var in cmd.get_envs().keys() { - let value = cmd.get_env(var).unwrap_or_default(); + for (var, value) in cmd.get_envs() { + let value = match value { + Some(s) => s, + None => continue, + }; self.env.insert( var.clone(), value