fix: improve usage of indicatif for tracking progress #101
| @ -10,13 +10,16 @@ use super::{ | ||||
| #[derive(Debug, Clone)] | ||||
| pub enum HarmonyEvent { | ||||
|     HarmonyStarted, | ||||
|     HarmonyFinished, | ||||
|     InterpretExecutionStarted { | ||||
|         execution_id: String, | ||||
|         topology: String, | ||||
|         interpret: String, | ||||
|         score: String, | ||||
|         message: String, | ||||
|     }, | ||||
|     InterpretExecutionFinished { | ||||
|         execution_id: String, | ||||
|         topology: String, | ||||
|         interpret: String, | ||||
|         score: String, | ||||
|  | ||||
| @ -5,6 +5,7 @@ use serde::Serialize; | ||||
| use serde_value::Value; | ||||
| 
 | ||||
| use super::{ | ||||
|     data::Id, | ||||
|     instrumentation::{self, HarmonyEvent}, | ||||
|     interpret::{Interpret, InterpretError, Outcome}, | ||||
|     inventory::Inventory, | ||||
| @ -20,9 +21,11 @@ pub trait Score<T: Topology>: | ||||
|         inventory: &Inventory, | ||||
|         topology: &T, | ||||
|     ) -> Result<Outcome, InterpretError> { | ||||
|         let id = Id::default(); | ||||
|         let interpret = self.create_interpret(); | ||||
| 
 | ||||
|         instrumentation::instrument(HarmonyEvent::InterpretExecutionStarted { | ||||
|             execution_id: id.clone().to_string(), | ||||
|             topology: topology.name().into(), | ||||
|             interpret: interpret.get_name().to_string(), | ||||
|             score: self.name(), | ||||
| @ -32,6 +35,7 @@ pub trait Score<T: Topology>: | ||||
|         let result = interpret.execute(inventory, topology).await; | ||||
| 
 | ||||
|         instrumentation::instrument(HarmonyEvent::InterpretExecutionFinished { | ||||
|             execution_id: id.clone().to_string(), | ||||
|             topology: topology.name().into(), | ||||
|             interpret: interpret.get_name().to_string(), | ||||
|             score: self.name(), | ||||
|  | ||||
| @ -59,7 +59,7 @@ impl<T: Topology + K8sclient + HelmCommand> Interpret<T> for ArgoInterpret { | ||||
|             .await | ||||
|             .unwrap(); | ||||
|         Ok(Outcome::success(format!( | ||||
|             "Successfully installed ArgoCD and {} Applications", | ||||
|             "ArgoCD installed with {} applications", | ||||
|             self.argo_apps.len() | ||||
|         ))) | ||||
|     } | ||||
|  | ||||
| @ -60,7 +60,7 @@ impl<A: Application, T: Topology + std::fmt::Debug> Interpret<T> for Application | ||||
|                 } | ||||
|             }; | ||||
|         } | ||||
|         Ok(Outcome::success("successfully created app".to_string())) | ||||
|         Ok(Outcome::success("Application created".to_string())) | ||||
|     } | ||||
| 
 | ||||
|     fn get_name(&self) -> InterpretName { | ||||
|  | ||||
| @ -46,7 +46,7 @@ where | ||||
|     } | ||||
| 
 | ||||
|     fn name(&self) -> String { | ||||
|         format!("Application: {}", self.application.name()) | ||||
|         format!("{} [ApplicationScore]", self.application.name()) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
|  | ||||
| @ -55,7 +55,7 @@ impl<T: Topology + HelmCommand> Score<T> for HelmChartScore { | ||||
|     } | ||||
| 
 | ||||
|     fn name(&self) -> String { | ||||
|         format!("{} {} HelmChartScore", self.release_name, self.chart_name) | ||||
|         format!("{} [HelmChartScore]", self.release_name) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -225,19 +225,20 @@ impl<T: Topology + HelmCommand> Interpret<T> for HelmChartInterpret { | ||||
|         match status { | ||||
|             helm_wrapper_rs::HelmDeployStatus::Deployed => Ok(Outcome::new( | ||||
|                 InterpretStatus::SUCCESS, | ||||
|                 "Helm Chart deployed".to_string(), | ||||
|                 format!("Helm Chart {} deployed", self.score.release_name), | ||||
|             )), | ||||
|             helm_wrapper_rs::HelmDeployStatus::PendingInstall => Ok(Outcome::new( | ||||
|                 InterpretStatus::RUNNING, | ||||
|                 "Helm Chart Pending install".to_string(), | ||||
|                 format!("Helm Chart {} pending install...", self.score.release_name), | ||||
|             )), | ||||
|             helm_wrapper_rs::HelmDeployStatus::PendingUpgrade => Ok(Outcome::new( | ||||
|                 InterpretStatus::RUNNING, | ||||
|                 "Helm Chart pending upgrade".to_string(), | ||||
|             )), | ||||
|             helm_wrapper_rs::HelmDeployStatus::Failed => Err(InterpretError::new( | ||||
|                 "Failed to install helm chart".to_string(), | ||||
|                 format!("Helm Chart {} pending upgrade...", self.score.release_name), | ||||
|             )), | ||||
|             helm_wrapper_rs::HelmDeployStatus::Failed => Err(InterpretError::new(format!( | ||||
|                 "Helm Chart {} installation failed", | ||||
|                 self.score.release_name | ||||
|             ))), | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|  | ||||
| @ -33,7 +33,10 @@ impl<T: Topology + PrometheusApplicationMonitoring<CRDPrometheus>> Score<T> | ||||
|     } | ||||
| 
 | ||||
|     fn name(&self) -> String { | ||||
|         "ApplicationMonitoringScore".to_string() | ||||
|         format!( | ||||
|             "{} monitoring [ApplicationMonitoringScore]", | ||||
|             self.application.name() | ||||
|         ) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -61,7 +64,9 @@ impl<T: Topology + PrometheusApplicationMonitoring<CRDPrometheus>> Interpret<T> | ||||
| 
 | ||||
|         match result { | ||||
|             Ok(outcome) => match outcome { | ||||
|                 PreparationOutcome::Success { details } => Ok(Outcome::success(details)), | ||||
|                 PreparationOutcome::Success { details: _ } => { | ||||
|                     Ok(Outcome::success("Prometheus installed".into())) | ||||
|                 } | ||||
|                 PreparationOutcome::Noop => Ok(Outcome::noop()), | ||||
|             }, | ||||
|             Err(err) => Err(InterpretError::from(err)), | ||||
|  | ||||
| @ -28,7 +28,7 @@ impl<T: Topology + HelmCommand + K8sclient> Score<T> for NtfyScore { | ||||
|     } | ||||
| 
 | ||||
|     fn name(&self) -> String { | ||||
|         "NtfyScore".to_string() | ||||
|         "alert receiver [NtfyScore]".into() | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -119,7 +119,7 @@ impl<T: Topology + HelmCommand + K8sclient> Interpret<T> for NtfyInterpret { | ||||
| 
 | ||||
|         debug!("exec into pod done"); | ||||
| 
 | ||||
|         Ok(Outcome::success("installed ntfy".to_string())) | ||||
|         Ok(Outcome::success("Ntfy installed".to_string())) | ||||
|     } | ||||
| 
 | ||||
|     fn get_name(&self) -> InterpretName { | ||||
|  | ||||
| @ -61,7 +61,7 @@ impl<T: Topology + K8sclient + PrometheusApplicationMonitoring<CRDPrometheus>> S | ||||
|     } | ||||
| 
 | ||||
|     fn name(&self) -> String { | ||||
|         "CRDApplicationAlertingScore".into() | ||||
|         "prometheus alerting [CRDAlertingScore]".into() | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -94,7 +94,7 @@ impl<T: Topology + K8sclient + PrometheusApplicationMonitoring<CRDPrometheus>> I | ||||
|         self.install_monitors(self.service_monitors.clone(), &client) | ||||
|             .await?; | ||||
|         Ok(Outcome::success( | ||||
|             "deployed application monitoring composants".to_string(), | ||||
|             "K8s monitoring components installed".to_string(), | ||||
|         )) | ||||
|     } | ||||
| 
 | ||||
|  | ||||
| @ -28,7 +28,7 @@ impl<T: Topology + TenantManager> Score<T> for TenantScore { | ||||
|     } | ||||
| 
 | ||||
|     fn name(&self) -> String { | ||||
|         format!("{} TenantScore", self.config.name) | ||||
|         format!("{} [TenantScore]", self.config.name) | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| @ -47,8 +47,8 @@ impl<T: Topology + TenantManager> Interpret<T> for TenantInterpret { | ||||
|         topology.provision_tenant(&self.tenant_config).await?; | ||||
| 
 | ||||
|         Ok(Outcome::success(format!( | ||||
|             "Successfully provisioned tenant {} with id {}", | ||||
|             self.tenant_config.name, self.tenant_config.id | ||||
|             "Tenant provisioned with id '{}'", | ||||
|             self.tenant_config.id | ||||
|         ))) | ||||
|     } | ||||
| 
 | ||||
|  | ||||
| @ -2,18 +2,15 @@ use harmony::{ | ||||
|     instrumentation::{self, HarmonyEvent}, | ||||
|     topology::TopologyStatus, | ||||
| }; | ||||
| use indicatif::{MultiProgress, ProgressBar}; | ||||
| use indicatif::MultiProgress; | ||||
| use indicatif_log_bridge::LogWrapper; | ||||
| use std::{ | ||||
|     collections::HashMap, | ||||
|     sync::{Arc, Mutex}, | ||||
| }; | ||||
| use std::sync::{Arc, Mutex}; | ||||
| 
 | ||||
| use crate::progress; | ||||
| use crate::progress::{IndicatifProgressTracker, ProgressTracker}; | ||||
| 
 | ||||
| pub fn init() -> tokio::task::JoinHandle<()> { | ||||
|     configure_logger(); | ||||
|     let handle = tokio::spawn(handle_events()); | ||||
|     let base_progress = configure_logger(); | ||||
|     let handle = tokio::spawn(handle_events(base_progress)); | ||||
| 
 | ||||
|     loop { | ||||
|         if instrumentation::instrument(HarmonyEvent::HarmonyStarted).is_ok() { | ||||
| @ -24,32 +21,45 @@ pub fn init() -> tokio::task::JoinHandle<()> { | ||||
|     handle | ||||
| } | ||||
| 
 | ||||
| fn configure_logger() { | ||||
| fn configure_logger() -> MultiProgress { | ||||
|     let logger = | ||||
|         env_logger::Builder::from_env(env_logger::Env::default().default_filter_or("info")).build(); | ||||
|     let level = logger.filter(); | ||||
|     let multi = MultiProgress::new(); | ||||
|     LogWrapper::new(multi.clone(), logger).try_init().unwrap(); | ||||
|     let progress = MultiProgress::new(); | ||||
| 
 | ||||
|     LogWrapper::new(progress.clone(), logger) | ||||
|         .try_init() | ||||
|         .unwrap(); | ||||
|     log::set_max_level(level); | ||||
| 
 | ||||
|     progress | ||||
| } | ||||
| 
 | ||||
| async fn handle_events() { | ||||
|     instrumentation::subscribe("Harmony CLI Logger", { | ||||
|         let sections: Arc<Mutex<HashMap<String, MultiProgress>>> = | ||||
|             Arc::new(Mutex::new(HashMap::new())); | ||||
|         let progress_bars: Arc<Mutex<HashMap<String, ProgressBar>>> = | ||||
|             Arc::new(Mutex::new(HashMap::new())); | ||||
| async fn handle_events(base_progress: MultiProgress) { | ||||
|     let progress_tracker = Arc::new(IndicatifProgressTracker::new(base_progress.clone())); | ||||
|     let preparing_topology = Arc::new(Mutex::new(false)); | ||||
|     let current_score: Arc<Mutex<Option<String>>> = Arc::new(Mutex::new(None)); | ||||
| 
 | ||||
|     instrumentation::subscribe("Harmony CLI Logger", { | ||||
|         move |event| { | ||||
|             let sections_clone = Arc::clone(§ions); | ||||
|             let progress_bars_clone = Arc::clone(&progress_bars); | ||||
|             let progress_tracker = Arc::clone(&progress_tracker); | ||||
|             let preparing_topology = Arc::clone(&preparing_topology); | ||||
|             let current_score = Arc::clone(¤t_score); | ||||
| 
 | ||||
|             async move { | ||||
|                 let mut sections = sections_clone.lock().unwrap(); | ||||
|                 let mut progress_bars = progress_bars_clone.lock().unwrap(); | ||||
|                 let mut preparing_topology = preparing_topology.lock().unwrap(); | ||||
|                 let mut current_score = current_score.lock().unwrap(); | ||||
| 
 | ||||
|                 match event { | ||||
|                     HarmonyEvent::HarmonyStarted => {} | ||||
|                     HarmonyEvent::HarmonyFinished => { | ||||
|                         progress_tracker.add_section( | ||||
|                             "harmony-summary", | ||||
|                             &format!("\n{} Harmony completed\n\n", crate::theme::EMOJI_HARMONY), | ||||
|                         ); | ||||
|                         progress_tracker.add_section("harmony-finished", "\n\n"); | ||||
|                         return false; | ||||
|                     } | ||||
|                     HarmonyEvent::TopologyStateChanged { | ||||
|                         topology, | ||||
|                         status, | ||||
| @ -60,111 +70,95 @@ async fn handle_events() { | ||||
|                         match status { | ||||
|                             TopologyStatus::Queued => {} | ||||
|                             TopologyStatus::Preparing => { | ||||
|                                 let section = progress::new_section(format!( | ||||
|                                     "{} Preparing environment: {topology}...", | ||||
|                                     crate::theme::EMOJI_TOPOLOGY, | ||||
|                                 )); | ||||
|                                 (*sections).insert(section_key, section); | ||||
|                                 progress_tracker.add_section( | ||||
|                                     §ion_key, | ||||
|                                     &format!( | ||||
|                                         "\n{} Preparing environment: {topology}...", | ||||
|                                         crate::theme::EMOJI_TOPOLOGY | ||||
|                                     ), | ||||
|                                 ); | ||||
|                                 (*preparing_topology) = true; | ||||
|                             } | ||||
|                             TopologyStatus::Success => { | ||||
|                                 let section = (*sections).get(§ion_key).unwrap(); | ||||
|                                 let progress = progress::add_spinner(section, "".into()); | ||||
| 
 | ||||
|                                 progress::success( | ||||
|                                     section, | ||||
|                                     Some(progress), | ||||
|                                     message.unwrap_or("".into()), | ||||
|                                 ); | ||||
| 
 | ||||
|                                 (*sections).remove(§ion_key); | ||||
|                                 (*preparing_topology) = false; | ||||
|                                 progress_tracker.add_task(§ion_key, "topology-success", ""); | ||||
|                                 progress_tracker | ||||
|                                     .finish_task("topology-success", &message.unwrap_or("".into())); | ||||
|                             } | ||||
|                             TopologyStatus::Noop => { | ||||
|                                 let section = (*sections).get(§ion_key).unwrap(); | ||||
|                                 let progress = progress::add_spinner(section, "".into()); | ||||
| 
 | ||||
|                                 progress::skip( | ||||
|                                     section, | ||||
|                                     Some(progress), | ||||
|                                     message.unwrap_or("".into()), | ||||
|                                 ); | ||||
| 
 | ||||
|                                 (*sections).remove(§ion_key); | ||||
|                                 (*preparing_topology) = false; | ||||
|                                 progress_tracker.add_task(§ion_key, "topology-skip", ""); | ||||
|                                 progress_tracker | ||||
|                                     .skip_task("topology-skip", &message.unwrap_or("".into())); | ||||
|                             } | ||||
|                             TopologyStatus::Error => { | ||||
|                                 let section = (*sections).get(§ion_key).unwrap(); | ||||
|                                 let progress = progress::add_spinner(section, "".into()); | ||||
| 
 | ||||
|                                 progress::error( | ||||
|                                     section, | ||||
|                                     Some(progress), | ||||
|                                     message.unwrap_or("".into()), | ||||
|                                 ); | ||||
| 
 | ||||
|                                 (*sections).remove(§ion_key); | ||||
|                                 progress_tracker.add_task(§ion_key, "topology-error", ""); | ||||
|                                 (*preparing_topology) = false; | ||||
|                                 progress_tracker | ||||
|                                     .fail_task("topology-error", &message.unwrap_or("".into())); | ||||
|                             } | ||||
|                         } | ||||
|                     } | ||||
|                     HarmonyEvent::InterpretExecutionStarted { | ||||
|                         execution_id: task_key, | ||||
|                         topology, | ||||
|                         interpret, | ||||
|                         interpret: _, | ||||
|                         score, | ||||
|                         message, | ||||
|                     } => { | ||||
|                         let section_key = if (*sections).contains_key(&topology_key(&topology)) { | ||||
|                         let is_key_topology = (*preparing_topology) | ||||
|                             && progress_tracker.contains_section(&topology_key(&topology)); | ||||
|                         let is_key_current_score = current_score.is_some() | ||||
|                             && progress_tracker | ||||
|                                 .contains_section(&score_key(¤t_score.clone().unwrap())); | ||||
|                         let is_key_score = progress_tracker.contains_section(&score_key(&score)); | ||||
| 
 | ||||
|                         let section_key = if is_key_topology { | ||||
|                             topology_key(&topology) | ||||
|                         } else if (*sections).contains_key(&score_key(&score)) { | ||||
|                             score_key(&interpret) | ||||
|                         } else if is_key_current_score { | ||||
|                             score_key(¤t_score.clone().unwrap()) | ||||
|                         } else if is_key_score { | ||||
|                             score_key(&score) | ||||
|                         } else { | ||||
|                             (*current_score) = Some(score.clone()); | ||||
|                             let key = score_key(&score); | ||||
|                             let section = progress::new_section(format!( | ||||
|                                 "\n{} Interpreting score: {score}...", | ||||
|                                 crate::theme::EMOJI_SCORE, | ||||
|                             )); | ||||
|                             (*sections).insert(key.clone(), section); | ||||
|                             progress_tracker.add_section( | ||||
|                                 &key, | ||||
|                                 &format!( | ||||
|                                     "{} Interpreting score: {score}...", | ||||
|                                     crate::theme::EMOJI_SCORE | ||||
|                                 ), | ||||
|                             ); | ||||
|                             key | ||||
|                         }; | ||||
|                         let section = (*sections).get(§ion_key).unwrap(); | ||||
|                         let progress_bar = progress::add_spinner(section, message); | ||||
| 
 | ||||
|                         (*progress_bars).insert(interpret_key(&interpret), progress_bar); | ||||
|                         progress_tracker.add_task(§ion_key, &task_key, &message); | ||||
|                     } | ||||
|                     HarmonyEvent::InterpretExecutionFinished { | ||||
|                         topology, | ||||
|                         interpret, | ||||
|                         execution_id: task_key, | ||||
|                         topology: _, | ||||
|                         interpret: _, | ||||
|                         score, | ||||
|                         outcome, | ||||
|                     } => { | ||||
|                         let has_topology = (*sections).contains_key(&topology_key(&topology)); | ||||
|                         let section_key = if has_topology { | ||||
|                             topology_key(&topology) | ||||
|                         } else { | ||||
|                             score_key(&score) | ||||
|                         }; | ||||
| 
 | ||||
|                         let section = (*sections).get(§ion_key).unwrap(); | ||||
|                         let progress_bar = | ||||
|                             (*progress_bars).get(&interpret_key(&interpret)).cloned(); | ||||
| 
 | ||||
|                         let _ = section.clear(); | ||||
|                         if current_score.is_some() && current_score.clone().unwrap() == score { | ||||
|                             (*current_score) = None; | ||||
|                         } | ||||
| 
 | ||||
|                         match outcome { | ||||
|                             Ok(outcome) => match outcome.status { | ||||
|                                 harmony::interpret::InterpretStatus::SUCCESS => { | ||||
|                                     progress::success(section, progress_bar, outcome.message) | ||||
|                                     progress_tracker.finish_task(&task_key, &outcome.message); | ||||
|                                 } | ||||
|                                 harmony::interpret::InterpretStatus::NOOP => { | ||||
|                                     progress::skip(section, progress_bar, outcome.message) | ||||
|                                     progress_tracker.skip_task(&task_key, &outcome.message); | ||||
|                                 } | ||||
|                                 _ => progress::error(section, progress_bar, outcome.message), | ||||
|                                 _ => progress_tracker.fail_task(&task_key, &outcome.message), | ||||
|                             }, | ||||
|                             Err(err) => { | ||||
|                                 progress::error(section, progress_bar, err.to_string()); | ||||
|                                 progress_tracker.fail_task(&task_key, &err.to_string()); | ||||
|                             } | ||||
|                         } | ||||
| 
 | ||||
|                         if !has_topology { | ||||
|                             (*progress_bars).remove(§ion_key); | ||||
|                         } | ||||
|                     } | ||||
|                 } | ||||
|                 true | ||||
| @ -181,7 +175,3 @@ fn topology_key(topology: &str) -> String { | ||||
| fn score_key(score: &str) -> String { | ||||
|     format!("score-{score}") | ||||
| } | ||||
| 
 | ||||
| fn interpret_key(interpret: &str) -> String { | ||||
|     format!("interpret-{interpret}") | ||||
| } | ||||
|  | ||||
| @ -1,5 +1,6 @@ | ||||
| use clap::Parser; | ||||
| use clap::builder::ArgPredicate; | ||||
| use harmony::instrumentation; | ||||
| use harmony::inventory::Inventory; | ||||
| use harmony::maestro::Maestro; | ||||
| use harmony::{score::Score, topology::Topology}; | ||||
| @ -97,6 +98,7 @@ pub async fn run<T: Topology + Send + Sync + 'static>( | ||||
| 
 | ||||
|     let result = init(maestro, args_struct).await; | ||||
| 
 | ||||
|     instrumentation::instrument(instrumentation::HarmonyEvent::HarmonyFinished).unwrap(); | ||||
|     let _ = tokio::try_join!(cli_logger_handle); | ||||
|     result | ||||
| } | ||||
|  | ||||
| @ -1,50 +1,163 @@ | ||||
| use indicatif::{MultiProgress, ProgressBar}; | ||||
| use std::collections::HashMap; | ||||
| use std::sync::{Arc, Mutex}; | ||||
| use std::time::Duration; | ||||
| 
 | ||||
| use indicatif::{MultiProgress, ProgressBar}; | ||||
| 
 | ||||
| pub fn new_section(title: String) -> MultiProgress { | ||||
|     let multi_progress = MultiProgress::new(); | ||||
|     let _ = multi_progress.println(title); | ||||
| 
 | ||||
|     multi_progress | ||||
| pub trait ProgressTracker: Send + Sync { | ||||
|     fn contains_section(&self, id: &str) -> bool; | ||||
|     fn add_section(&self, id: &str, message: &str); | ||||
|     fn add_task(&self, section_id: &str, task_id: &str, message: &str); | ||||
|     fn finish_task(&self, id: &str, message: &str); | ||||
|     fn fail_task(&self, id: &str, message: &str); | ||||
|     fn skip_task(&self, id: &str, message: &str); | ||||
|     fn clear(&self); | ||||
| } | ||||
| 
 | ||||
| pub fn add_spinner(multi_progress: &MultiProgress, message: String) -> ProgressBar { | ||||
|     let progress = multi_progress.add(ProgressBar::new_spinner()); | ||||
| 
 | ||||
|     progress.set_style(crate::theme::SPINNER_STYLE.clone()); | ||||
|     progress.set_message(message); | ||||
|     progress.enable_steady_tick(Duration::from_millis(100)); | ||||
| 
 | ||||
|     progress | ||||
| struct Section { | ||||
|     header_index: usize, | ||||
|     task_count: usize, | ||||
|     pb: ProgressBar, | ||||
| } | ||||
| 
 | ||||
| pub fn success(multi_progress: &MultiProgress, progress: Option<ProgressBar>, message: String) { | ||||
|     if let Some(progress) = progress { | ||||
|         multi_progress.remove(&progress) | ||||
| struct IndicatifProgressTrackerState { | ||||
|     sections: HashMap<String, Section>, | ||||
|     tasks: HashMap<String, ProgressBar>, | ||||
|     pb_count: usize, | ||||
| } | ||||
| 
 | ||||
| #[derive(Clone)] | ||||
| pub struct IndicatifProgressTracker { | ||||
|     mp: MultiProgress, | ||||
|     state: Arc<Mutex<IndicatifProgressTrackerState>>, | ||||
| } | ||||
| 
 | ||||
| impl IndicatifProgressTracker { | ||||
|     pub fn new(base: MultiProgress) -> Self { | ||||
|         // The indicatif log bridge will insert a progress bar at the top.
 | ||||
|         // To prevent our first section from being erased, we need to create
 | ||||
|         // a dummy progress bar as our first progress bar.
 | ||||
|         let _ = base.clear(); | ||||
|         let log_pb = base.add(ProgressBar::new(1)); | ||||
| 
 | ||||
|         let mut sections = HashMap::new(); | ||||
|         sections.insert( | ||||
|             "__log__".into(), | ||||
|             Section { | ||||
|                 header_index: 0, | ||||
|                 task_count: 0, | ||||
|                 pb: log_pb.clone(), | ||||
|             }, | ||||
|         ); | ||||
| 
 | ||||
|         let mut tasks = HashMap::new(); | ||||
|         tasks.insert("__log__".into(), log_pb); | ||||
| 
 | ||||
|         let state = Arc::new(Mutex::new(IndicatifProgressTrackerState { | ||||
|             sections, | ||||
|             tasks, | ||||
|             pb_count: 1, | ||||
|         })); | ||||
| 
 | ||||
|         Self { mp: base, state } | ||||
|     } | ||||
| } | ||||
| 
 | ||||
| impl ProgressTracker for IndicatifProgressTracker { | ||||
|     fn add_section(&self, id: &str, message: &str) { | ||||
|         let mut state = self.state.lock().unwrap(); | ||||
| 
 | ||||
|         let header_pb = self | ||||
|             .mp | ||||
|             .add(ProgressBar::new(1).with_style(crate::theme::SECTION_STYLE.clone())); | ||||
|         header_pb.finish_with_message(message.to_string()); | ||||
| 
 | ||||
|         let header_index = state.pb_count; | ||||
|         state.pb_count += 1; | ||||
| 
 | ||||
|         state.sections.insert( | ||||
|             id.to_string(), | ||||
|             Section { | ||||
|                 header_index, | ||||
|                 task_count: 0, | ||||
|                 pb: header_pb, | ||||
|             }, | ||||
|         ); | ||||
|     } | ||||
| 
 | ||||
|     let progress = multi_progress.add(ProgressBar::new_spinner()); | ||||
|     progress.set_style(crate::theme::SUCCESS_SPINNER_STYLE.clone()); | ||||
|     progress.finish_with_message(message); | ||||
| } | ||||
|     fn add_task(&self, section_id: &str, task_id: &str, message: &str) { | ||||
|         let mut state = self.state.lock().unwrap(); | ||||
| 
 | ||||
| pub fn error(multi_progress: &MultiProgress, progress: Option<ProgressBar>, message: String) { | ||||
|     if let Some(progress) = progress { | ||||
|         multi_progress.remove(&progress) | ||||
|         let insertion_index = { | ||||
|             let current_section = state | ||||
|                 .sections | ||||
|                 .get(section_id) | ||||
|                 .expect("Section ID not found"); | ||||
|             current_section.header_index + current_section.task_count + 1 // +1 to insert after header
 | ||||
|         }; | ||||
| 
 | ||||
|         let pb = self.mp.insert(insertion_index, ProgressBar::new_spinner()); | ||||
|         pb.set_style(crate::theme::SPINNER_STYLE.clone()); | ||||
|         pb.set_prefix("  "); | ||||
|         pb.set_message(message.to_string()); | ||||
|         pb.enable_steady_tick(Duration::from_millis(80)); | ||||
| 
 | ||||
|         state.pb_count += 1; | ||||
| 
 | ||||
|         let section = state | ||||
|             .sections | ||||
|             .get_mut(section_id) | ||||
|             .expect("Section ID not found"); | ||||
|         section.task_count += 1; | ||||
| 
 | ||||
|         // We inserted a new progress bar, so we must update the header_index
 | ||||
|         // for all subsequent sections.
 | ||||
|         for (id, s) in state.sections.iter_mut() { | ||||
|             if id != section_id && s.header_index >= insertion_index { | ||||
|                 s.header_index += 1; | ||||
|             } | ||||
|         } | ||||
| 
 | ||||
|     let progress = multi_progress.add(ProgressBar::new_spinner()); | ||||
|     progress.set_style(crate::theme::ERROR_SPINNER_STYLE.clone()); | ||||
|     progress.finish_with_message(message); | ||||
| } | ||||
| 
 | ||||
| pub fn skip(multi_progress: &MultiProgress, progress: Option<ProgressBar>, message: String) { | ||||
|     if let Some(progress) = progress { | ||||
|         multi_progress.remove(&progress) | ||||
|         state.tasks.insert(task_id.to_string(), pb); | ||||
|     } | ||||
| 
 | ||||
|     let progress = multi_progress.add(ProgressBar::new_spinner()); | ||||
|     progress.set_style(crate::theme::SKIP_SPINNER_STYLE.clone()); | ||||
|     progress.finish_with_message(message); | ||||
|     fn finish_task(&self, id: &str, message: &str) { | ||||
|         let state = self.state.lock().unwrap(); | ||||
|         if let Some(pb) = state.tasks.get(id) { | ||||
|             pb.set_style(crate::theme::SUCCESS_SPINNER_STYLE.clone()); | ||||
|             pb.finish_with_message(message.to_string()); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fn fail_task(&self, id: &str, message: &str) { | ||||
|         let state = self.state.lock().unwrap(); | ||||
|         if let Some(pb) = state.tasks.get(id) { | ||||
|             pb.set_style(crate::theme::ERROR_SPINNER_STYLE.clone()); | ||||
|             pb.finish_with_message(message.to_string()); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fn skip_task(&self, id: &str, message: &str) { | ||||
|         let state = self.state.lock().unwrap(); | ||||
|         if let Some(pb) = state.tasks.get(id) { | ||||
|             pb.set_style(crate::theme::SKIP_SPINNER_STYLE.clone()); | ||||
|             pb.finish_with_message(message.to_string()); | ||||
|         } | ||||
|     } | ||||
| 
 | ||||
|     fn contains_section(&self, id: &str) -> bool { | ||||
|         let state = self.state.lock().unwrap(); | ||||
|         state.sections.contains_key(id) | ||||
|     } | ||||
| 
 | ||||
|     fn clear(&self) { | ||||
|         let mut state = self.state.lock().unwrap(); | ||||
| 
 | ||||
|         state.tasks.values().for_each(|p| self.mp.remove(p)); | ||||
|         state.tasks.clear(); | ||||
|         state.sections.values().for_each(|s| self.mp.remove(&s.pb)); | ||||
|         state.sections.clear(); | ||||
|         state.pb_count = 0; | ||||
| 
 | ||||
|         let _ = self.mp.clear(); | ||||
|     } | ||||
| } | ||||
|  | ||||
| @ -11,8 +11,11 @@ pub static EMOJI_TOPOLOGY: Emoji<'_, '_> = Emoji("📦", ""); | ||||
| pub static EMOJI_SCORE: Emoji<'_, '_> = Emoji("🎶", ""); | ||||
| 
 | ||||
| lazy_static! { | ||||
|     pub static ref SECTION_STYLE: ProgressStyle = ProgressStyle::default_spinner() | ||||
|         .template("{wide_msg:.bold}") | ||||
|         .unwrap(); | ||||
|     pub static ref SPINNER_STYLE: ProgressStyle = ProgressStyle::default_spinner() | ||||
|         .template("    {spinner:.green} {msg}") | ||||
|         .template("    {spinner:.green} {wide_msg}") | ||||
|         .unwrap() | ||||
|         .tick_strings(&["⠋", "⠙", "⠹", "⠸", "⠼", "⠴", "⠦", "⠧", "⠇", "⠏"]); | ||||
|     pub static ref SUCCESS_SPINNER_STYLE: ProgressStyle = SPINNER_STYLE | ||||
|  | ||||
| @ -1,10 +1,7 @@ | ||||
| use indicatif::{MultiProgress, ProgressBar}; | ||||
| use indicatif_log_bridge::LogWrapper; | ||||
| use harmony_cli::progress::{IndicatifProgressTracker, ProgressTracker}; | ||||
| use indicatif::MultiProgress; | ||||
| use log::error; | ||||
| use std::{ | ||||
|     collections::HashMap, | ||||
|     sync::{Arc, Mutex}, | ||||
| }; | ||||
| use std::sync::Arc; | ||||
| 
 | ||||
| use crate::instrumentation::{self, HarmonyComposerEvent}; | ||||
| 
 | ||||
| @ -22,85 +19,57 @@ pub fn init() -> tokio::task::JoinHandle<()> { | ||||
| } | ||||
| 
 | ||||
| fn configure_logger() { | ||||
|     let logger = | ||||
|     env_logger::Builder::from_env(env_logger::Env::default().default_filter_or("info")).build(); | ||||
|     let level = logger.filter(); | ||||
|     let multi = MultiProgress::new(); | ||||
|     LogWrapper::new(multi.clone(), logger).try_init().unwrap(); | ||||
|     log::set_max_level(level); | ||||
| } | ||||
| 
 | ||||
| pub async fn handle_events() { | ||||
|     const PROGRESS_SETUP: &str = "project-initialization"; | ||||
|     let progress_tracker = Arc::new(IndicatifProgressTracker::new(MultiProgress::new())); | ||||
| 
 | ||||
|     const SETUP_SECTION: &str = "project-initialization"; | ||||
|     const COMPILTATION_TASK: &str = "compilation"; | ||||
|     const PROGRESS_DEPLOYMENT: &str = "deployment"; | ||||
| 
 | ||||
|     instrumentation::subscribe("Harmony Composer Logger", { | ||||
|         let progresses: Arc<Mutex<HashMap<String, MultiProgress>>> = | ||||
|             Arc::new(Mutex::new(HashMap::new())); | ||||
|         let compilation_progress = Arc::new(Mutex::new(None::<ProgressBar>)); | ||||
| 
 | ||||
|         move |event| { | ||||
|             let progresses_clone = Arc::clone(&progresses); | ||||
|             let compilation_progress_clone = Arc::clone(&compilation_progress); | ||||
|             let progress_tracker = Arc::clone(&progress_tracker); | ||||
| 
 | ||||
|             async move { | ||||
|                 let mut progresses_guard = progresses_clone.lock().unwrap(); | ||||
|                 let mut compilation_progress_guard = compilation_progress_clone.lock().unwrap(); | ||||
| 
 | ||||
|                 match event { | ||||
|                     HarmonyComposerEvent::HarmonyComposerStarted => {} | ||||
|                     HarmonyComposerEvent::ProjectInitializationStarted => { | ||||
|                         let multi_progress = harmony_cli::progress::new_section(format!( | ||||
|                         progress_tracker.add_section( | ||||
|                             SETUP_SECTION, | ||||
|                             &format!( | ||||
|                                 "{} Initializing Harmony project...", | ||||
|                                 harmony_cli::theme::EMOJI_HARMONY, | ||||
|                         )); | ||||
|                         (*progresses_guard).insert(PROGRESS_SETUP.to_string(), multi_progress); | ||||
|                             ), | ||||
|                         ); | ||||
|                     } | ||||
|                     HarmonyComposerEvent::ProjectInitialized => println!("\n"), | ||||
|                     HarmonyComposerEvent::ProjectInitialized => {} | ||||
|                     HarmonyComposerEvent::ProjectCompilationStarted { details } => { | ||||
|                         let initialization_progress = | ||||
|                             (*progresses_guard).get(PROGRESS_SETUP).unwrap(); | ||||
|                         let _ = initialization_progress.clear(); | ||||
| 
 | ||||
|                         let progress = | ||||
|                             harmony_cli::progress::add_spinner(initialization_progress, details); | ||||
|                         *compilation_progress_guard = Some(progress); | ||||
|                         progress_tracker.add_task(SETUP_SECTION, COMPILTATION_TASK, &details); | ||||
|                     } | ||||
|                     HarmonyComposerEvent::ProjectCompiled => { | ||||
|                         let initialization_progress = | ||||
|                             (*progresses_guard).get(PROGRESS_SETUP).unwrap(); | ||||
| 
 | ||||
|                         harmony_cli::progress::success( | ||||
|                             initialization_progress, | ||||
|                             (*compilation_progress_guard).take(), | ||||
|                             "project compiled".to_string(), | ||||
|                         ); | ||||
|                         progress_tracker.finish_task(COMPILTATION_TASK, "project compiled"); | ||||
|                     } | ||||
|                     HarmonyComposerEvent::ProjectCompilationFailed { details } => { | ||||
|                         let initialization_progress = | ||||
|                             (*progresses_guard).get(PROGRESS_SETUP).unwrap(); | ||||
| 
 | ||||
|                         harmony_cli::progress::error( | ||||
|                             initialization_progress, | ||||
|                             (*compilation_progress_guard).take(), | ||||
|                             "failed to compile project".to_string(), | ||||
|                         ); | ||||
|                         progress_tracker.fail_task(COMPILTATION_TASK, "failed to compile project"); | ||||
| 
 | ||||
|                         error!("{details}"); | ||||
|                     } | ||||
|                     HarmonyComposerEvent::DeploymentStarted { target } => { | ||||
|                         let multi_progress = harmony_cli::progress::new_section(format!( | ||||
|                             "{} Starting deployment to {target}...\n\n", | ||||
|                             harmony_cli::theme::EMOJI_DEPLOY | ||||
|                         )); | ||||
|                         (*progresses_guard).insert(PROGRESS_DEPLOYMENT.to_string(), multi_progress); | ||||
|                         progress_tracker.add_section( | ||||
|                             PROGRESS_DEPLOYMENT, | ||||
|                             &format!( | ||||
|                                 "\n{} Deploying project to {target}...\n", | ||||
|                                 harmony_cli::theme::EMOJI_DEPLOY, | ||||
|                             ), | ||||
|                         ); | ||||
|                     } | ||||
|                     HarmonyComposerEvent::DeploymentCompleted => { | ||||
|                         progress_tracker.clear(); | ||||
|                     } | ||||
|                     HarmonyComposerEvent::DeploymentCompleted => println!("\n"), | ||||
|                     HarmonyComposerEvent::Shutdown => { | ||||
|                         for (_, progresses) in (*progresses_guard).iter() { | ||||
|                             progresses.clear().unwrap(); | ||||
|                         } | ||||
| 
 | ||||
|                         return false; | ||||
|                     } | ||||
|                 } | ||||
|  | ||||
		Loading…
	
		Reference in New Issue
	
	Block a user