From 2d4cf4474fa7ee9fc4d1537e324f5a7bb04718dd Mon Sep 17 00:00:00 2001 From: Federico Guerinoni Date: Wed, 30 Nov 2022 12:39:10 +0100 Subject: [PATCH 1/4] feat: Wait env ready after `project new` This allow the user to get more feedback during the creation of a prject. Signed-off-by: Federico Guerinoni --- Cargo.lock | 25 +++++++++++++++++++++++++ cargo-shuttle/Cargo.toml | 1 + cargo-shuttle/src/lib.rs | 35 +++++++++++++++++++++++++++++++++-- 3 files changed, 59 insertions(+), 2 deletions(-) diff --git a/Cargo.lock b/Cargo.lock index cf5a5f084..8fd854818 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -1433,6 +1433,7 @@ dependencies = [ "git2", "headers", "ignore", + "indicatif", "indoc", "log", "openssl", @@ -3290,6 +3291,18 @@ dependencies = [ "serde", ] +[[package]] +name = "indicatif" +version = "0.17.2" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "4295cbb7573c16d310e99e713cf9e75101eb190ab31fccd35f2d2691b4352b19" +dependencies = [ + "console", + "number_prefix", + "portable-atomic", + "unicode-width", +] + [[package]] name = "indoc" version = "1.0.7" @@ -3949,6 +3962,12 @@ dependencies = [ "libc", ] +[[package]] +name = "number_prefix" +version = "0.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "830b246a0e5f20af87141b25c173cd1b609bd7779a4617d6ec582abaf90870f3" + [[package]] name = "objc" version = "0.2.7" @@ -4421,6 +4440,12 @@ dependencies = [ "universal-hash", ] +[[package]] +name = "portable-atomic" +version = "0.3.15" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "15eb2c6e362923af47e13c23ca5afb859e83d54452c55b0b9ac763b8f7c1ac16" + [[package]] name = "portpicker" version = "0.1.1" diff --git a/cargo-shuttle/Cargo.toml b/cargo-shuttle/Cargo.toml index 30e286c84..ce4b92f43 100644 --- a/cargo-shuttle/Cargo.toml +++ b/cargo-shuttle/Cargo.toml @@ -25,6 +25,7 @@ flate2 = "1.0.25" futures = "0.3.25" git2 = "0.14.2" headers = "0.3.8" +indicatif = "0.17.2" ignore = "0.4.18" indoc = "1.0.7" log = "0.4.17" diff --git a/cargo-shuttle/src/lib.rs b/cargo-shuttle/src/lib.rs index 4dfd06701..bb491a012 100644 --- a/cargo-shuttle/src/lib.rs +++ b/cargo-shuttle/src/lib.rs @@ -27,7 +27,7 @@ use futures::StreamExt; use git2::{Repository, StatusOptions}; use ignore::overrides::OverrideBuilder; use ignore::WalkBuilder; -use shuttle_common::models::secret; +use shuttle_common::models::{project, secret}; use shuttle_service::loader::{build_crate, Loader}; use shuttle_service::Logger; use std::fmt::Write; @@ -485,7 +485,38 @@ impl Shuttle { } async fn project_create(&self, client: &Client) -> Result<()> { - let project = client.create_project(self.ctx.project_name()).await?; + let pb = indicatif::ProgressBar::new_spinner(); + pb.enable_steady_tick(std::time::Duration::from_millis(100)); + pb.set_style( + indicatif::ProgressStyle::with_template("{spinner:.blue} {msg}") + .unwrap() + .tick_strings(&[ + "( ● )", + "( ● )", + "( ● )", + "( ● )", + "( ●)", + "( ● )", + "( ● )", + "( ● )", + "( ● )", + "(● )", + "(●●●●●●)", + ]), + ); + + let mut project = client.create_project(self.ctx.project_name()).await?; + + loop { + if project.state == project::State::Ready { + break; + } + + pb.set_message(format!("{project}")); + project = client.get_project(self.ctx.project_name()).await?; + } + + pb.finish_with_message("Done"); println!("{project}"); From 714ddf53bdbd37777f23648e292a9835a69810bd Mon Sep 17 00:00:00 2001 From: Federico Guerinoni Date: Wed, 30 Nov 2022 15:16:58 +0100 Subject: [PATCH 2/4] feat: Add --follow argument to `project status` command This allow the user to see update of a project command, while is creating or removing. Signed-off-by: Federico Guerinoni --- cargo-shuttle/src/args.rs | 6 ++++- cargo-shuttle/src/lib.rs | 47 +++++++++++++++++++++++++++++++++++---- 2 files changed, 48 insertions(+), 5 deletions(-) diff --git a/cargo-shuttle/src/args.rs b/cargo-shuttle/src/args.rs index d096914e7..7a2c783cb 100644 --- a/cargo-shuttle/src/args.rs +++ b/cargo-shuttle/src/args.rs @@ -112,7 +112,11 @@ pub enum ProjectCommand { /// remove this project environment from shuttle Rm, /// show the status of this project's environment on shuttle - Status, + Status { + #[clap(short, long)] + /// Follow status of project command + follow: bool, + }, } #[derive(Parser, Clone, Debug)] diff --git a/cargo-shuttle/src/lib.rs b/cargo-shuttle/src/lib.rs index bb491a012..87d0b2fb1 100644 --- a/cargo-shuttle/src/lib.rs +++ b/cargo-shuttle/src/lib.rs @@ -93,7 +93,9 @@ impl Shuttle { Command::Secrets => self.secrets(&client).await, Command::Auth(auth_args) => self.auth(auth_args, &client).await, Command::Project(ProjectCommand::New) => self.project_create(&client).await, - Command::Project(ProjectCommand::Status) => self.project_status(&client).await, + Command::Project(ProjectCommand::Status { follow }) => { + self.project_status(&client, follow).await + } Command::Project(ProjectCommand::Rm) => self.project_delete(&client).await, _ => { unreachable!("commands that don't need a client have already been matched") @@ -523,11 +525,48 @@ impl Shuttle { Ok(()) } - async fn project_status(&self, client: &Client) -> Result<()> { - let project = client.get_project(self.ctx.project_name()).await?; + async fn project_status(&self, client: &Client, follow: bool) -> Result<()> { + let mut project = client.get_project(self.ctx.project_name()).await?; + + match follow { + true => { + let pb = indicatif::ProgressBar::new_spinner(); + pb.enable_steady_tick(std::time::Duration::from_millis(100)); + pb.set_style( + indicatif::ProgressStyle::with_template("{spinner:.blue} {msg}") + .unwrap() + .tick_strings(&[ + "( ● )", + "( ● )", + "( ● )", + "( ● )", + "( ●)", + "( ● )", + "( ● )", + "( ● )", + "( ● )", + "(● )", + "(●●●●●●)", + ]), + ); - println!("{project}"); + loop { + if project.state == project::State::Ready + || project.state == project::State::Destroyed + { + break; + } + + pb.set_message(format!("{project}")); + project = client.get_project(self.ctx.project_name()).await?; + } + pb.finish_with_message("Done"); + } + false => (), + } + + println!("{project}"); Ok(()) } From 53ae1651adb2bf2f3474689b4892ea78f0c0a7fb Mon Sep 17 00:00:00 2001 From: Federico Guerinoni Date: Wed, 30 Nov 2022 18:23:51 +0100 Subject: [PATCH 3/4] refactor: Unify in same function the code for wait with spinner Now it is possible to pass to a function a client function to call and a slice of status to wait while a spinner run on CLI. Signed-off-by: Federico Guerinoni --- cargo-shuttle/src/lib.rs | 103 ++++++++++++++++++--------------------- 1 file changed, 48 insertions(+), 55 deletions(-) diff --git a/cargo-shuttle/src/lib.rs b/cargo-shuttle/src/lib.rs index 87d0b2fb1..40e3a11ff 100644 --- a/cargo-shuttle/src/lib.rs +++ b/cargo-shuttle/src/lib.rs @@ -4,6 +4,7 @@ pub mod config; mod factory; mod init; +use shuttle_common::project::ProjectName; use std::collections::BTreeMap; use std::ffi::OsString; use std::fs::{read_to_string, File}; @@ -487,10 +488,53 @@ impl Shuttle { } async fn project_create(&self, client: &Client) -> Result<()> { + self.wait_with_spinner( + &[project::State::Ready], + Client::create_project, + self.ctx.project_name(), + client, + ) + .await?; + + Ok(()) + } + + async fn project_status(&self, client: &Client, follow: bool) -> Result<()> { + match follow { + true => { + self.wait_with_spinner( + &[project::State::Ready, project::State::Destroyed], + Client::get_project, + self.ctx.project_name(), + client, + ) + .await?; + } + false => { + let project = client.get_project(self.ctx.project_name()).await?; + println!("{project}"); + } + } + + Ok(()) + } + + async fn wait_with_spinner<'a, F, Fut>( + &self, + states_to_check: &[project::State], + f: F, + project_name: &'a ProjectName, + client: &'a Client, + ) -> Result<(), anyhow::Error> + where + F: Fn(&'a Client, &'a ProjectName) -> Fut, + Fut: std::future::Future> + 'a, + { + let mut project = f(client, project_name).await?; let pb = indicatif::ProgressBar::new_spinner(); - pb.enable_steady_tick(std::time::Duration::from_millis(100)); + pb.enable_steady_tick(std::time::Duration::from_millis(350)); pb.set_style( - indicatif::ProgressStyle::with_template("{spinner:.blue} {msg}") + indicatif::ProgressStyle::with_template("{spinner:.orange} {msg}") .unwrap() .tick_strings(&[ "( ● )", @@ -506,66 +550,15 @@ impl Shuttle { "(●●●●●●)", ]), ); - - let mut project = client.create_project(self.ctx.project_name()).await?; - loop { - if project.state == project::State::Ready { + if states_to_check.contains(&project.state) { break; } pb.set_message(format!("{project}")); - project = client.get_project(self.ctx.project_name()).await?; + project = client.get_project(project_name).await?; } - pb.finish_with_message("Done"); - - println!("{project}"); - - Ok(()) - } - - async fn project_status(&self, client: &Client, follow: bool) -> Result<()> { - let mut project = client.get_project(self.ctx.project_name()).await?; - - match follow { - true => { - let pb = indicatif::ProgressBar::new_spinner(); - pb.enable_steady_tick(std::time::Duration::from_millis(100)); - pb.set_style( - indicatif::ProgressStyle::with_template("{spinner:.blue} {msg}") - .unwrap() - .tick_strings(&[ - "( ● )", - "( ● )", - "( ● )", - "( ● )", - "( ●)", - "( ● )", - "( ● )", - "( ● )", - "( ● )", - "(● )", - "(●●●●●●)", - ]), - ); - - loop { - if project.state == project::State::Ready - || project.state == project::State::Destroyed - { - break; - } - - pb.set_message(format!("{project}")); - project = client.get_project(self.ctx.project_name()).await?; - } - - pb.finish_with_message("Done"); - } - false => (), - } - println!("{project}"); Ok(()) } From 77b0218d237737c61bcad7e97c2984a32d80efb6 Mon Sep 17 00:00:00 2001 From: oddgrd <29732646+oddgrd@users.noreply.github.com> Date: Wed, 7 Dec 2022 14:46:15 +0100 Subject: [PATCH 4/4] feat: add project errored state --- cargo-shuttle/src/lib.rs | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/cargo-shuttle/src/lib.rs b/cargo-shuttle/src/lib.rs index 40e3a11ff..0405a8a18 100644 --- a/cargo-shuttle/src/lib.rs +++ b/cargo-shuttle/src/lib.rs @@ -489,7 +489,7 @@ impl Shuttle { async fn project_create(&self, client: &Client) -> Result<()> { self.wait_with_spinner( - &[project::State::Ready], + &[project::State::Ready, project::state::Errored], Client::create_project, self.ctx.project_name(), client, @@ -503,7 +503,11 @@ impl Shuttle { match follow { true => { self.wait_with_spinner( - &[project::State::Ready, project::State::Destroyed], + &[ + project::State::Ready, + project::State::Destroyed, + project::state::Errored, + ], Client::get_project, self.ctx.project_name(), client,