diff --git a/src/cargo/core/compiler/job_queue/mod.rs b/src/cargo/core/compiler/job_queue/mod.rs index 7ab12a3a265..b04e6a42bae 100644 --- a/src/cargo/core/compiler/job_queue/mod.rs +++ b/src/cargo/core/compiler/job_queue/mod.rs @@ -854,7 +854,7 @@ impl<'gctx> DrainState<'gctx> { } fn handle_error( - &self, + &mut self, shell: &mut Shell, err_state: &mut ErrorsDuringDrain, new_err: impl Into, @@ -863,6 +863,7 @@ impl<'gctx> DrainState<'gctx> { if new_err.print_always || err_state.count == 0 { crate::display_error(&new_err.error, shell); if err_state.count == 0 && !self.active.is_empty() { + self.progress.indicate_error(); let _ = shell.warn("build failed, waiting for other jobs to finish..."); } err_state.count += 1; diff --git a/src/cargo/util/context/mod.rs b/src/cargo/util/context/mod.rs index 9824eff9017..06e9dc477bf 100644 --- a/src/cargo/util/context/mod.rs +++ b/src/cargo/util/context/mod.rs @@ -2749,6 +2749,7 @@ pub struct TermConfig { pub struct ProgressConfig { pub when: ProgressWhen, pub width: Option, + pub taskbar: Option, } #[derive(Debug, Default, Deserialize)] @@ -2781,10 +2782,12 @@ where "auto" => Ok(Some(ProgressConfig { when: ProgressWhen::Auto, width: None, + taskbar: None, })), "never" => Ok(Some(ProgressConfig { when: ProgressWhen::Never, width: None, + taskbar: None, })), "always" => Err(E::custom("\"always\" progress requires a `width` key")), _ => Err(E::unknown_variant(s, &["auto", "never"])), @@ -2806,6 +2809,7 @@ where if let ProgressConfig { when: ProgressWhen::Always, width: None, + .. } = pc { return Err(serde::de::Error::custom( diff --git a/src/cargo/util/progress.rs b/src/cargo/util/progress.rs index 655fabcd4d9..3d28c36315c 100644 --- a/src/cargo/util/progress.rs +++ b/src/cargo/util/progress.rs @@ -74,6 +74,113 @@ struct Format { style: ProgressStyle, max_width: usize, max_print: usize, + taskbar: TaskbarProgress, +} + +/// Taskbar progressbar +/// +/// Outputs ANSI sequences according to the `Operating system commands`. +struct TaskbarProgress { + enabled: bool, + error: bool, +} + +/// A taskbar progress value printable as ANSI OSC escape code +enum TaskbarValue { + /// Do not output anything + None, + /// Remove progress + Remove, + /// Progress value 0-100 + Value(f64), + /// Indeterminate state (no bar, just animation) + Indeterminate, + /// Progress value 0-100 in the error state + Error(f64), +} + +enum ProgressOutput { + /// Print progress without a message + PrintNow, + /// Progress, message and taskbar progress + TextAndTaskbar(String, TaskbarValue), + /// Only taskbar progress, no message and no text progress + Taskbar(TaskbarValue), +} + +impl TaskbarProgress { + #[cfg(test)] + fn new(enabled: bool) -> Self { + Self { + enabled, + error: false, + } + } + + /// Creates a new `TaskbarProgress` from a cargo's config. + /// If not explicitly enabled or disabled, detect a supported terminal. + fn from_config(gctx: &GlobalContext) -> Self { + let enabled = match gctx.progress_config().taskbar { + Some(v) => v, + None => { + gctx.get_env("WT_SESSION").is_ok() + || gctx.get_env("ConEmuANSI").ok() == Some("ON".into()) + } + }; + + Self { + enabled, + error: false, + } + } + + pub fn remove(&self) -> TaskbarValue { + if self.enabled { + TaskbarValue::Remove + } else { + TaskbarValue::None + } + } + + pub fn value(&self, percent: f64) -> TaskbarValue { + match (self.enabled, self.error) { + (true, false) => TaskbarValue::Value(percent), + (true, true) => TaskbarValue::Error(percent), + (false, _) => TaskbarValue::None, + } + } + + pub fn indeterminate(&self) -> TaskbarValue { + match (self.enabled, self.error) { + (true, false) => TaskbarValue::Indeterminate, + (true, true) => TaskbarValue::Error(100.0), + (false, _) => TaskbarValue::None, + } + } + + pub fn error(&mut self) { + self.error = true; + } +} + +impl std::fmt::Display for TaskbarValue { + fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { + // From https://conemu.github.io/en/AnsiEscapeCodes.html#ConEmu_specific_OSC + // ESC ] 9 ; 4 ; st ; pr ST + // When st is 0: remove progress. + // When st is 1: set progress value to pr (number, 0-100). + // When st is 2: set error state in taskbar, pr is optional. + // When st is 3: set indeterminate state, pr is ignored. + // When st is 4: set paused state, pr is optional. + let (state, progress) = match self { + Self::None => return Ok(()), + Self::Remove => (0, 0.0), + Self::Value(v) => (1, *v), + Self::Indeterminate => (3, 0.0), + Self::Error(v) => (2, *v), + }; + write!(f, "\x1b]9;4;{state};{progress:.0}\x1b\\") + } } impl<'gctx> Progress<'gctx> { @@ -126,6 +233,7 @@ impl<'gctx> Progress<'gctx> { // 50 gives some space for text after the progress bar, // even on narrow (e.g. 80 char) terminals. max_print: 50, + taskbar: TaskbarProgress::from_config(gctx), }, name: name.to_string(), done: false, @@ -223,7 +331,7 @@ impl<'gctx> Progress<'gctx> { /// calling it too often. pub fn print_now(&mut self, msg: &str) -> CargoResult<()> { match &mut self.state { - Some(s) => s.print("", msg), + Some(s) => s.print(ProgressOutput::PrintNow, msg), None => Ok(()), } } @@ -234,6 +342,13 @@ impl<'gctx> Progress<'gctx> { s.clear(); } } + + /// Sets the taskbar progress to the error state. + pub fn indicate_error(&mut self) { + if let Some(s) = &mut self.state { + s.format.taskbar.error() + } + } } impl Throttle { @@ -269,6 +384,7 @@ impl Throttle { impl<'gctx> State<'gctx> { fn tick(&mut self, cur: usize, max: usize, msg: &str) -> CargoResult<()> { if self.done { + write!(self.gctx.shell().err(), "{}", self.format.taskbar.remove())?; return Ok(()); } @@ -280,21 +396,30 @@ impl<'gctx> State<'gctx> { // return back to the beginning of the line for the next print. self.try_update_max_width(); if let Some(pbar) = self.format.progress(cur, max) { - self.print(&pbar, msg)?; + self.print(pbar, msg)?; } Ok(()) } - fn print(&mut self, prefix: &str, msg: &str) -> CargoResult<()> { + fn print(&mut self, progress: ProgressOutput, msg: &str) -> CargoResult<()> { self.throttle.update(); self.try_update_max_width(); + let (mut line, taskbar) = match progress { + ProgressOutput::PrintNow => (String::new(), None), + ProgressOutput::TextAndTaskbar(prefix, taskbar_value) => (prefix, Some(taskbar_value)), + ProgressOutput::Taskbar(taskbar_value) => (String::new(), Some(taskbar_value)), + }; + // make sure we have enough room for the header if self.format.max_width < 15 { + // even if we don't have space we can still output taskbar progress + if let Some(tb) = taskbar { + write!(self.gctx.shell().err(), "{}\r", tb)?; + } return Ok(()); } - let mut line = prefix.to_string(); self.format.render(&mut line, msg); while line.len() < self.format.max_width - 15 { line.push(' '); @@ -305,7 +430,11 @@ impl<'gctx> State<'gctx> { let mut shell = self.gctx.shell(); shell.set_needs_clear(false); shell.status_header(&self.name)?; - write!(shell.err(), "{}\r", line)?; + if let Some(tb) = taskbar { + write!(shell.err(), "{}{}\r", line, tb)?; + } else { + write!(shell.err(), "{}\r", line)?; + } self.last_line = Some(line); shell.set_needs_clear(true); } @@ -314,6 +443,8 @@ impl<'gctx> State<'gctx> { } fn clear(&mut self) { + // Always clear the taskbar progress + let _ = write!(self.gctx.shell().err(), "{}", self.format.taskbar.remove()); // No need to clear if the progress is not currently being displayed. if self.last_line.is_some() && !self.gctx.shell().is_cleared() { self.gctx.shell().err_erase_line(); @@ -331,7 +462,7 @@ impl<'gctx> State<'gctx> { } impl Format { - fn progress(&self, cur: usize, max: usize) -> Option { + fn progress(&self, cur: usize, max: usize) -> Option { assert!(cur <= max); // Render the percentage at the far right and then figure how long the // progress bar is @@ -342,8 +473,16 @@ impl Format { ProgressStyle::Ratio => format!(" {}/{}", cur, max), ProgressStyle::Indeterminate => String::new(), }; + let taskbar = match self.style { + ProgressStyle::Percentage | ProgressStyle::Ratio => self.taskbar.value(pct * 100.0), + ProgressStyle::Indeterminate => self.taskbar.indeterminate(), + }; + let extra_len = stats.len() + 2 /* [ and ] */ + 15 /* status header */; let Some(display_width) = self.width().checked_sub(extra_len) else { + if self.taskbar.enabled { + return Some(ProgressOutput::Taskbar(taskbar)); + } return None; }; @@ -371,7 +510,7 @@ impl Format { string.push(']'); string.push_str(&stats); - Some(string) + Some(ProgressOutput::TextAndTaskbar(string, taskbar)) } fn render(&self, string: &mut String, msg: &str) { @@ -398,7 +537,11 @@ impl Format { #[cfg(test)] fn progress_status(&self, cur: usize, max: usize, msg: &str) -> Option { - let mut ret = self.progress(cur, max)?; + let mut ret = match self.progress(cur, max)? { + // Check only the variant that contains text + ProgressOutput::TextAndTaskbar(text, _) => text, + _ => return None, + }; self.render(&mut ret, msg); Some(ret) } @@ -420,6 +563,7 @@ fn test_progress_status() { style: ProgressStyle::Ratio, max_print: 40, max_width: 60, + taskbar: TaskbarProgress::new(false), }; assert_eq!( format.progress_status(0, 4, ""), @@ -493,6 +637,7 @@ fn test_progress_status_percentage() { style: ProgressStyle::Percentage, max_print: 40, max_width: 60, + taskbar: TaskbarProgress::new(false), }; assert_eq!( format.progress_status(0, 77, ""), @@ -518,6 +663,7 @@ fn test_progress_status_too_short() { style: ProgressStyle::Percentage, max_print: 25, max_width: 25, + taskbar: TaskbarProgress::new(false), }; assert_eq!( format.progress_status(1, 1, ""), @@ -528,6 +674,17 @@ fn test_progress_status_too_short() { style: ProgressStyle::Percentage, max_print: 24, max_width: 24, + taskbar: TaskbarProgress::new(false), }; assert_eq!(format.progress_status(1, 1, ""), None); } + +#[test] +fn test_taskbar_disabled() { + let taskbar = TaskbarProgress::new(false); + let mut out = String::new(); + out.push_str(&taskbar.remove().to_string()); + out.push_str(&taskbar.value(10.0).to_string()); + out.push_str(&taskbar.indeterminate().to_string()); + assert!(out.is_empty()); +} diff --git a/src/doc/src/reference/config.md b/src/doc/src/reference/config.md index 11ac2f9a5dc..1f40676a108 100644 --- a/src/doc/src/reference/config.md +++ b/src/doc/src/reference/config.md @@ -191,13 +191,14 @@ metadata_key1 = "value" metadata_key2 = "value" [term] -quiet = false # whether cargo output is quiet -verbose = false # whether cargo provides verbose output -color = 'auto' # whether cargo colorizes output -hyperlinks = true # whether cargo inserts links into output -unicode = true # whether cargo can render output using non-ASCII unicode characters -progress.when = 'auto' # whether cargo shows progress bar -progress.width = 80 # width of progress bar +quiet = false # whether cargo output is quiet +verbose = false # whether cargo provides verbose output +color = 'auto' # whether cargo colorizes output +hyperlinks = true # whether cargo inserts links into output +unicode = true # whether cargo can render output using non-ASCII unicode characters +progress.when = 'auto' # whether cargo shows progress bar +progress.width = 80 # width of progress bar +progress.taskbar = true # whether cargo reports progress to terminal emulator ``` ## Environment variables @@ -1361,6 +1362,13 @@ Controls whether or not progress bar is shown in the terminal. Possible values: Sets the width for progress bar. +#### `term.progress.taskbar` +* Type: bool +* Default: auto-detect +* Environment: `CARGO_TERM_PROGRESS_TASKBAR` + +Report progess to the teminal emulator for display in places like the task bar. + [`cargo bench`]: ../commands/cargo-bench.md [`cargo login`]: ../commands/cargo-login.md [`cargo logout`]: ../commands/cargo-logout.md