fix: make sure demo works on both local & remote target #107
| @ -7,9 +7,11 @@ use harmony::{ | |||||||
|             ApplicationScore, RustWebFramework, RustWebapp, |             ApplicationScore, RustWebFramework, RustWebapp, | ||||||
|             features::{ContinuousDelivery, Monitoring}, |             features::{ContinuousDelivery, Monitoring}, | ||||||
|         }, |         }, | ||||||
|  |         load_balancer::LoadBalancerScore, | ||||||
|         monitoring::alert_channel::{ |         monitoring::alert_channel::{ | ||||||
|             discord_alert_channel::DiscordWebhook, webhook_receiver::WebhookReceiver, |             discord_alert_channel::DiscordWebhook, webhook_receiver::WebhookReceiver, | ||||||
|         }, |         }, | ||||||
|  |         okd::bootstrap_load_balancer::OKDBootstrapLoadBalancerScore, | ||||||
|     }, |     }, | ||||||
|     topology::{K8sAnywhereTopology, Url}, |     topology::{K8sAnywhereTopology, Url}, | ||||||
| }; | }; | ||||||
|  | |||||||
| @ -2,6 +2,8 @@ use log::debug; | |||||||
| use once_cell::sync::Lazy; | use once_cell::sync::Lazy; | ||||||
| use tokio::sync::broadcast; | use tokio::sync::broadcast; | ||||||
| 
 | 
 | ||||||
|  | use crate::modules::application::ApplicationFeatureStatus; | ||||||
|  | 
 | ||||||
| use super::{ | use super::{ | ||||||
|     interpret::{InterpretError, Outcome}, |     interpret::{InterpretError, Outcome}, | ||||||
|     topology::TopologyStatus, |     topology::TopologyStatus, | ||||||
| @ -30,6 +32,12 @@ pub enum HarmonyEvent { | |||||||
|         status: TopologyStatus, |         status: TopologyStatus, | ||||||
|         message: Option<String>, |         message: Option<String>, | ||||||
|     }, |     }, | ||||||
|  |     ApplicationFeatureStateChanged { | ||||||
|  |         topology: String, | ||||||
|  |         application: String, | ||||||
|  |         feature: String, | ||||||
|  |         status: ApplicationFeatureStatus, | ||||||
|  |     }, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| static HARMONY_EVENT_BUS: Lazy<broadcast::Sender<HarmonyEvent>> = Lazy::new(|| { | static HARMONY_EVENT_BUS: Lazy<broadcast::Sender<HarmonyEvent>> = Lazy::new(|| { | ||||||
|  | |||||||
| @ -120,7 +120,7 @@ impl K8sClient { | |||||||
|                     .expect("Couldn't unwrap status"); |                     .expect("Couldn't unwrap status"); | ||||||
| 
 | 
 | ||||||
|                 if let Some(s) = status.status { |                 if let Some(s) = status.status { | ||||||
|                     debug!("Status: {}", s); |                     debug!("Status: {} - {:?}", s, status.details); | ||||||
|                     if s == "Success" { Ok(()) } else { Err(s) } |                     if s == "Success" { Ok(()) } else { Err(s) } | ||||||
|                 } else { |                 } else { | ||||||
|                     Err("Couldn't get inner status of pod exec".to_string()) |                     Err("Couldn't get inner status of pod exec".to_string()) | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| use std::{io::Write, process::Command, sync::Arc}; | use std::{io::Write, process::Command, sync::Arc}; | ||||||
| 
 | 
 | ||||||
| use async_trait::async_trait; | use async_trait::async_trait; | ||||||
| use log::{debug, error}; | use log::info; | ||||||
| use serde_yaml::Value; | use serde_yaml::Value; | ||||||
| use tempfile::NamedTempFile; | use tempfile::NamedTempFile; | ||||||
| 
 | 
 | ||||||
| @ -56,14 +56,11 @@ impl<A: OCICompliant + HelmPackage> ContinuousDelivery<A> { | |||||||
|         chart_url: String, |         chart_url: String, | ||||||
|         image_name: String, |         image_name: String, | ||||||
|     ) -> Result<(), String> { |     ) -> Result<(), String> { | ||||||
|         error!( |         // TODO: This works only with local k3d installations, which is fine only for current demo purposes. We assume usage of K8sAnywhereTopology"
 | ||||||
|             "FIXME This works only with local k3d installations, which is fine only for current demo purposes. We assume usage of K8sAnywhereTopology" |         // https://git.nationtech.io/NationTech/harmony/issues/106
 | ||||||
|         ); |  | ||||||
| 
 |  | ||||||
|         error!("TODO hardcoded k3d bin path is wrong"); |  | ||||||
|         let k3d_bin_path = (*HARMONY_DATA_DIR).join("k3d").join("k3d"); |         let k3d_bin_path = (*HARMONY_DATA_DIR).join("k3d").join("k3d"); | ||||||
|         // --- 1. Import the container image into the k3d cluster ---
 |         // --- 1. Import the container image into the k3d cluster ---
 | ||||||
|         debug!( |         info!( | ||||||
|             "Importing image '{}' into k3d cluster 'harmony'", |             "Importing image '{}' into k3d cluster 'harmony'", | ||||||
|             image_name |             image_name | ||||||
|         ); |         ); | ||||||
| @ -80,7 +77,7 @@ impl<A: OCICompliant + HelmPackage> ContinuousDelivery<A> { | |||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         // --- 2. Get the kubeconfig for the k3d cluster and write it to a temp file ---
 |         // --- 2. Get the kubeconfig for the k3d cluster and write it to a temp file ---
 | ||||||
|         debug!("Retrieving kubeconfig for k3d cluster 'harmony'"); |         info!("Retrieving kubeconfig for k3d cluster 'harmony'"); | ||||||
|         let kubeconfig_output = Command::new(&k3d_bin_path) |         let kubeconfig_output = Command::new(&k3d_bin_path) | ||||||
|             .args(["kubeconfig", "get", "harmony"]) |             .args(["kubeconfig", "get", "harmony"]) | ||||||
|             .output() |             .output() | ||||||
| @ -101,7 +98,7 @@ impl<A: OCICompliant + HelmPackage> ContinuousDelivery<A> { | |||||||
|         let kubeconfig_path = temp_kubeconfig.path().to_str().unwrap(); |         let kubeconfig_path = temp_kubeconfig.path().to_str().unwrap(); | ||||||
| 
 | 
 | ||||||
|         // --- 3. Install or upgrade the Helm chart in the cluster ---
 |         // --- 3. Install or upgrade the Helm chart in the cluster ---
 | ||||||
|         debug!( |         info!( | ||||||
|             "Deploying Helm chart '{}' to namespace '{}'", |             "Deploying Helm chart '{}' to namespace '{}'", | ||||||
|             chart_url, app_name |             chart_url, app_name | ||||||
|         ); |         ); | ||||||
| @ -131,7 +128,7 @@ impl<A: OCICompliant + HelmPackage> ContinuousDelivery<A> { | |||||||
|             )); |             )); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         debug!("Successfully deployed '{}' to local k3d cluster.", app_name); |         info!("Successfully deployed '{}' to local k3d cluster.", app_name); | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
| } | } | ||||||
| @ -151,14 +148,12 @@ impl< | |||||||
|         // Or ask for it when unknown
 |         // Or ask for it when unknown
 | ||||||
| 
 | 
 | ||||||
|         let helm_chart = self.application.build_push_helm_package(&image).await?; |         let helm_chart = self.application.build_push_helm_package(&image).await?; | ||||||
|         debug!("Pushed new helm chart {helm_chart}"); |  | ||||||
| 
 | 
 | ||||||
|         error!("TODO Make building image configurable/skippable if image already exists (prompt)"); |         // TODO: Make building image configurable/skippable if image already exists (prompt)")
 | ||||||
|  |         // https://git.nationtech.io/NationTech/harmony/issues/104
 | ||||||
|         let image = self.application.build_push_oci_image().await?; |         let image = self.application.build_push_oci_image().await?; | ||||||
|         debug!("Pushed new docker image {image}"); |  | ||||||
| 
 | 
 | ||||||
|         debug!("Installing ContinuousDelivery feature"); |         // TODO: this is a temporary hack for demo purposes, the deployment target should be driven
 | ||||||
|         // TODO this is a temporary hack for demo purposes, the deployment target should be driven
 |  | ||||||
|         // by the topology only and we should not have to know how to perform tasks like this for
 |         // by the topology only and we should not have to know how to perform tasks like this for
 | ||||||
|         // which the topology should be responsible.
 |         // which the topology should be responsible.
 | ||||||
|         //
 |         //
 | ||||||
| @ -171,17 +166,20 @@ impl< | |||||||
|         // access it. This forces every Topology to understand the concept of targets though... So
 |         // access it. This forces every Topology to understand the concept of targets though... So
 | ||||||
|         // instead I'll create a new Capability which is MultiTargetTopology and we'll see how it
 |         // instead I'll create a new Capability which is MultiTargetTopology and we'll see how it
 | ||||||
|         // goes. It still does not feel right though.
 |         // goes. It still does not feel right though.
 | ||||||
|  |         //
 | ||||||
|  |         // https://git.nationtech.io/NationTech/harmony/issues/106
 | ||||||
|         match topology.current_target() { |         match topology.current_target() { | ||||||
|             DeploymentTarget::LocalDev => { |             DeploymentTarget::LocalDev => { | ||||||
|  |                 info!("Deploying {} locally...", self.application.name()); | ||||||
|                 self.deploy_to_local_k3d(self.application.name(), helm_chart, image) |                 self.deploy_to_local_k3d(self.application.name(), helm_chart, image) | ||||||
|                     .await?; |                     .await?; | ||||||
|             } |             } | ||||||
|             target => { |             target => { | ||||||
|                 debug!("Deploying to target {target:?}"); |                 info!("Deploying {} to target {target:?}", self.application.name()); | ||||||
|                 let score = ArgoHelmScore { |                 let score = ArgoHelmScore { | ||||||
|                     namespace: "harmonydemo-staging".to_string(), |                     namespace: "harmony-example-rust-webapp".to_string(), | ||||||
|                     openshift: false, |                     openshift: true, | ||||||
|                     domain: "argo.harmonydemo.apps.st.mcd".to_string(), |                     domain: "argo.harmonydemo.apps.ncd0.harmony.mcd".to_string(), | ||||||
|                     argo_apps: vec![ArgoApplication::from(CDApplicationConfig { |                     argo_apps: vec![ArgoApplication::from(CDApplicationConfig { | ||||||
|                         // helm pull oci://hub.nationtech.io/harmony/harmony-example-rust-webapp-chart --version 0.1.0
 |                         // helm pull oci://hub.nationtech.io/harmony/harmony-example-rust-webapp-chart --version 0.1.0
 | ||||||
|                         version: Version::from("0.1.0").unwrap(), |                         version: Version::from("0.1.0").unwrap(), | ||||||
| @ -189,7 +187,7 @@ impl< | |||||||
|                         helm_chart_name: "harmony-example-rust-webapp-chart".to_string(), |                         helm_chart_name: "harmony-example-rust-webapp-chart".to_string(), | ||||||
|                         values_overrides: None, |                         values_overrides: None, | ||||||
|                         name: "harmony-demo-rust-webapp".to_string(), |                         name: "harmony-demo-rust-webapp".to_string(), | ||||||
|                         namespace: "harmonydemo-staging".to_string(), |                         namespace: "harmony-example-rust-webapp".to_string(), | ||||||
|                     })], |                     })], | ||||||
|                 }; |                 }; | ||||||
|                 score |                 score | ||||||
|  | |||||||
| @ -1,5 +1,4 @@ | |||||||
| use async_trait::async_trait; | use async_trait::async_trait; | ||||||
| use log::error; |  | ||||||
| use non_blank_string_rs::NonBlankString; | use non_blank_string_rs::NonBlankString; | ||||||
| use serde::Serialize; | use serde::Serialize; | ||||||
| use std::str::FromStr; | use std::str::FromStr; | ||||||
| @ -50,7 +49,6 @@ impl<T: Topology + K8sclient + HelmCommand> Interpret<T> for ArgoInterpret { | |||||||
|         inventory: &Inventory, |         inventory: &Inventory, | ||||||
|         topology: &T, |         topology: &T, | ||||||
|     ) -> Result<Outcome, InterpretError> { |     ) -> Result<Outcome, InterpretError> { | ||||||
|         error!("Uncomment below, only disabled for debugging"); |  | ||||||
|         self.score.interpret(inventory, topology).await?; |         self.score.interpret(inventory, topology).await?; | ||||||
| 
 | 
 | ||||||
|         let k8s_client = topology.k8s_client().await?; |         let k8s_client = topology.k8s_client().await?; | ||||||
| @ -58,9 +56,14 @@ impl<T: Topology + K8sclient + HelmCommand> Interpret<T> for ArgoInterpret { | |||||||
|             .apply_yaml_many(&self.argo_apps.iter().map(|a| a.to_yaml()).collect(), None) |             .apply_yaml_many(&self.argo_apps.iter().map(|a| a.to_yaml()).collect(), None) | ||||||
|             .await |             .await | ||||||
|             .unwrap(); |             .unwrap(); | ||||||
|  | 
 | ||||||
|         Ok(Outcome::success(format!( |         Ok(Outcome::success(format!( | ||||||
|             "ArgoCD installed with {} applications", |             "ArgoCD installed with {} {}", | ||||||
|             self.argo_apps.len() |             self.argo_apps.len(), | ||||||
|  |             match self.argo_apps.len() { | ||||||
|  |                 1 => "application", | ||||||
|  |                 _ => "applications", | ||||||
|  |             } | ||||||
|         ))) |         ))) | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -4,6 +4,7 @@ use crate::modules::application::{Application, ApplicationFeature}; | |||||||
| use crate::modules::monitoring::application_monitoring::application_monitoring_score::ApplicationMonitoringScore; | use crate::modules::monitoring::application_monitoring::application_monitoring_score::ApplicationMonitoringScore; | ||||||
| use crate::modules::monitoring::kube_prometheus::crd::crd_alertmanager_config::CRDPrometheus; | use crate::modules::monitoring::kube_prometheus::crd::crd_alertmanager_config::CRDPrometheus; | ||||||
| 
 | 
 | ||||||
|  | use crate::topology::MultiTargetTopology; | ||||||
| use crate::{ | use crate::{ | ||||||
|     inventory::Inventory, |     inventory::Inventory, | ||||||
|     modules::monitoring::{ |     modules::monitoring::{ | ||||||
| @ -33,6 +34,7 @@ impl< | |||||||
|         + 'static |         + 'static | ||||||
|         + TenantManager |         + TenantManager | ||||||
|         + K8sclient |         + K8sclient | ||||||
|  |         + MultiTargetTopology | ||||||
|         + std::fmt::Debug |         + std::fmt::Debug | ||||||
|         + PrometheusApplicationMonitoring<CRDPrometheus>, |         + PrometheusApplicationMonitoring<CRDPrometheus>, | ||||||
| > ApplicationFeature<T> for Monitoring | > ApplicationFeature<T> for Monitoring | ||||||
| @ -55,11 +57,11 @@ impl< | |||||||
|         }; |         }; | ||||||
|         let ntfy = NtfyScore { |         let ntfy = NtfyScore { | ||||||
|             namespace: namespace.clone(), |             namespace: namespace.clone(), | ||||||
|             host: "localhost".to_string(), |             host: "ntfy.harmonydemo.apps.ncd0.harmony.mcd".to_string(), | ||||||
|         }; |         }; | ||||||
|         ntfy.interpret(&Inventory::empty(), topology) |         ntfy.interpret(&Inventory::empty(), topology) | ||||||
|             .await |             .await | ||||||
|             .expect("couldn't create interpret for ntfy"); |             .map_err(|e| e.to_string())?; | ||||||
| 
 | 
 | ||||||
|         let ntfy_default_auth_username = "harmony"; |         let ntfy_default_auth_username = "harmony"; | ||||||
|         let ntfy_default_auth_password = "harmony"; |         let ntfy_default_auth_password = "harmony"; | ||||||
| @ -96,7 +98,7 @@ impl< | |||||||
|         alerting_score |         alerting_score | ||||||
|             .interpret(&Inventory::empty(), topology) |             .interpret(&Inventory::empty(), topology) | ||||||
|             .await |             .await | ||||||
|             .unwrap(); |             .map_err(|e| e.to_string())?; | ||||||
|         Ok(()) |         Ok(()) | ||||||
|     } |     } | ||||||
|     fn name(&self) -> String { |     fn name(&self) -> String { | ||||||
|  | |||||||
| @ -14,11 +14,19 @@ use serde::Serialize; | |||||||
| 
 | 
 | ||||||
| use crate::{ | use crate::{ | ||||||
|     data::{Id, Version}, |     data::{Id, Version}, | ||||||
|  |     instrumentation::{self, HarmonyEvent}, | ||||||
|     interpret::{Interpret, InterpretError, InterpretName, InterpretStatus, Outcome}, |     interpret::{Interpret, InterpretError, InterpretName, InterpretStatus, Outcome}, | ||||||
|     inventory::Inventory, |     inventory::Inventory, | ||||||
|     topology::Topology, |     topology::Topology, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
|  | #[derive(Clone, Debug)] | ||||||
|  | pub enum ApplicationFeatureStatus { | ||||||
|  |     Installing, | ||||||
|  |     Installed, | ||||||
|  |     Failed { details: String }, | ||||||
|  | } | ||||||
|  | 
 | ||||||
| pub trait Application: std::fmt::Debug + Send + Sync { | pub trait Application: std::fmt::Debug + Send + Sync { | ||||||
|     fn name(&self) -> String; |     fn name(&self) -> String; | ||||||
| } | } | ||||||
| @ -47,13 +55,34 @@ impl<A: Application, T: Topology + std::fmt::Debug> Interpret<T> for Application | |||||||
|                 .join(", ") |                 .join(", ") | ||||||
|         ); |         ); | ||||||
|         for feature in self.features.iter() { |         for feature in self.features.iter() { | ||||||
|             debug!( |             instrumentation::instrument(HarmonyEvent::ApplicationFeatureStateChanged { | ||||||
|                 "Installing feature {} for application {app_name}", |                 topology: topology.name().into(), | ||||||
|                 feature.name() |                 application: self.application.name(), | ||||||
|             ); |                 feature: feature.name(), | ||||||
|  |                 status: ApplicationFeatureStatus::Installing, | ||||||
|  |             }) | ||||||
|  |             .unwrap(); | ||||||
|  | 
 | ||||||
|             let _ = match feature.ensure_installed(topology).await { |             let _ = match feature.ensure_installed(topology).await { | ||||||
|                 Ok(()) => (), |                 Ok(()) => { | ||||||
|  |                     instrumentation::instrument(HarmonyEvent::ApplicationFeatureStateChanged { | ||||||
|  |                         topology: topology.name().into(), | ||||||
|  |                         application: self.application.name(), | ||||||
|  |                         feature: feature.name(), | ||||||
|  |                         status: ApplicationFeatureStatus::Installed, | ||||||
|  |                     }) | ||||||
|  |                     .unwrap(); | ||||||
|  |                 } | ||||||
|                 Err(msg) => { |                 Err(msg) => { | ||||||
|  |                     instrumentation::instrument(HarmonyEvent::ApplicationFeatureStateChanged { | ||||||
|  |                         topology: topology.name().into(), | ||||||
|  |                         application: self.application.name(), | ||||||
|  |                         feature: feature.name(), | ||||||
|  |                         status: ApplicationFeatureStatus::Failed { | ||||||
|  |                             details: msg.clone(), | ||||||
|  |                         }, | ||||||
|  |                     }) | ||||||
|  |                     .unwrap(); | ||||||
|                     return Err(InterpretError::new(format!( |                     return Err(InterpretError::new(format!( | ||||||
|                         "Application Interpret failed to install feature : {msg}" |                         "Application Interpret failed to install feature : {msg}" | ||||||
|                     ))); |                     ))); | ||||||
|  | |||||||
| @ -10,7 +10,7 @@ use dockerfile_builder::Dockerfile; | |||||||
| use dockerfile_builder::instruction::{CMD, COPY, ENV, EXPOSE, FROM, RUN, USER, WORKDIR}; | use dockerfile_builder::instruction::{CMD, COPY, ENV, EXPOSE, FROM, RUN, USER, WORKDIR}; | ||||||
| use dockerfile_builder::instruction_builder::CopyBuilder; | use dockerfile_builder::instruction_builder::CopyBuilder; | ||||||
| use futures_util::StreamExt; | use futures_util::StreamExt; | ||||||
| use log::{debug, error, log_enabled}; | use log::{debug, info, log_enabled}; | ||||||
| use serde::Serialize; | use serde::Serialize; | ||||||
| use tar::Archive; | use tar::Archive; | ||||||
| 
 | 
 | ||||||
| @ -73,19 +73,19 @@ impl Application for RustWebapp { | |||||||
| #[async_trait] | #[async_trait] | ||||||
| impl HelmPackage for RustWebapp { | impl HelmPackage for RustWebapp { | ||||||
|     async fn build_push_helm_package(&self, image_url: &str) -> Result<String, String> { |     async fn build_push_helm_package(&self, image_url: &str) -> Result<String, String> { | ||||||
|         debug!("Starting Helm chart build and push for '{}'", self.name); |         info!("Starting Helm chart build and push for '{}'", self.name); | ||||||
| 
 | 
 | ||||||
|         // 1. Create the Helm chart files on disk.
 |         // 1. Create the Helm chart files on disk.
 | ||||||
|         let chart_dir = self |         let chart_dir = self | ||||||
|             .create_helm_chart_files(image_url) |             .create_helm_chart_files(image_url) | ||||||
|             .map_err(|e| format!("Failed to create Helm chart files: {}", e))?; |             .map_err(|e| format!("Failed to create Helm chart files: {}", e))?; | ||||||
|         debug!("Successfully created Helm chart files in {:?}", chart_dir); |         info!("Successfully created Helm chart files in {:?}", chart_dir); | ||||||
| 
 | 
 | ||||||
|         // 2. Package the chart into a .tgz archive.
 |         // 2. Package the chart into a .tgz archive.
 | ||||||
|         let packaged_chart_path = self |         let packaged_chart_path = self | ||||||
|             .package_helm_chart(&chart_dir) |             .package_helm_chart(&chart_dir) | ||||||
|             .map_err(|e| format!("Failed to package Helm chart: {}", e))?; |             .map_err(|e| format!("Failed to package Helm chart: {}", e))?; | ||||||
|         debug!( |         info!( | ||||||
|             "Successfully packaged Helm chart: {}", |             "Successfully packaged Helm chart: {}", | ||||||
|             packaged_chart_path.to_string_lossy() |             packaged_chart_path.to_string_lossy() | ||||||
|         ); |         ); | ||||||
| @ -94,7 +94,7 @@ impl HelmPackage for RustWebapp { | |||||||
|         let oci_chart_url = self |         let oci_chart_url = self | ||||||
|             .push_helm_chart(&packaged_chart_path) |             .push_helm_chart(&packaged_chart_path) | ||||||
|             .map_err(|e| format!("Failed to push Helm chart: {}", e))?; |             .map_err(|e| format!("Failed to push Helm chart: {}", e))?; | ||||||
|         debug!("Successfully pushed Helm chart to: {}", oci_chart_url); |         info!("Successfully pushed Helm chart to: {}", oci_chart_url); | ||||||
| 
 | 
 | ||||||
|         Ok(oci_chart_url) |         Ok(oci_chart_url) | ||||||
|     } |     } | ||||||
| @ -107,20 +107,20 @@ impl OCICompliant for RustWebapp { | |||||||
|     async fn build_push_oci_image(&self) -> Result<String, String> { |     async fn build_push_oci_image(&self) -> Result<String, String> { | ||||||
|         // This function orchestrates the build and push process.
 |         // This function orchestrates the build and push process.
 | ||||||
|         // It's async to match the trait definition, though the underlying docker commands are blocking.
 |         // It's async to match the trait definition, though the underlying docker commands are blocking.
 | ||||||
|         debug!("Starting OCI image build and push for '{}'", self.name); |         info!("Starting OCI image build and push for '{}'", self.name); | ||||||
| 
 | 
 | ||||||
|         // 1. Build the image by calling the synchronous helper function.
 |         // 1. Build the image by calling the synchronous helper function.
 | ||||||
|         let image_tag = self.image_name(); |         let image_tag = self.image_name(); | ||||||
|         self.build_docker_image(&image_tag) |         self.build_docker_image(&image_tag) | ||||||
|             .await |             .await | ||||||
|             .map_err(|e| format!("Failed to build Docker image: {}", e))?; |             .map_err(|e| format!("Failed to build Docker image: {}", e))?; | ||||||
|         debug!("Successfully built Docker image: {}", image_tag); |         info!("Successfully built Docker image: {}", image_tag); | ||||||
| 
 | 
 | ||||||
|         // 2. Push the image to the registry.
 |         // 2. Push the image to the registry.
 | ||||||
|         self.push_docker_image(&image_tag) |         self.push_docker_image(&image_tag) | ||||||
|             .await |             .await | ||||||
|             .map_err(|e| format!("Failed to push Docker image: {}", e))?; |             .map_err(|e| format!("Failed to push Docker image: {}", e))?; | ||||||
|         debug!("Successfully pushed Docker image to: {}", image_tag); |         info!("Successfully pushed Docker image to: {}", image_tag); | ||||||
| 
 | 
 | ||||||
|         Ok(image_tag) |         Ok(image_tag) | ||||||
|     } |     } | ||||||
| @ -195,7 +195,7 @@ impl RustWebapp { | |||||||
|         ); |         ); | ||||||
| 
 | 
 | ||||||
|         while let Some(msg) = image_build_stream.next().await { |         while let Some(msg) = image_build_stream.next().await { | ||||||
|             println!("Message: {msg:?}"); |             debug!("Message: {msg:?}"); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         Ok(image_name.to_string()) |         Ok(image_name.to_string()) | ||||||
| @ -219,7 +219,7 @@ impl RustWebapp { | |||||||
|         ); |         ); | ||||||
| 
 | 
 | ||||||
|         while let Some(msg) = push_image_stream.next().await { |         while let Some(msg) = push_image_stream.next().await { | ||||||
|             println!("Message: {msg:?}"); |             debug!("Message: {msg:?}"); | ||||||
|         } |         } | ||||||
| 
 | 
 | ||||||
|         Ok(image_tag.to_string()) |         Ok(image_tag.to_string()) | ||||||
| @ -288,9 +288,8 @@ impl RustWebapp { | |||||||
|                         .unwrap(), |                         .unwrap(), | ||||||
|                 ); |                 ); | ||||||
|                 // Copy the compiled binary from the builder stage.
 |                 // Copy the compiled binary from the builder stage.
 | ||||||
|                 error!( |                 // TODO: Should not be using score name here, instead should use name from Cargo.toml
 | ||||||
|                     "FIXME Should not be using score name here, instead should use name from Cargo.toml" |                 // https://git.nationtech.io/NationTech/harmony/issues/105
 | ||||||
|                 ); |  | ||||||
|                 let binary_path_in_builder = format!("/app/target/release/{}", self.name); |                 let binary_path_in_builder = format!("/app/target/release/{}", self.name); | ||||||
|                 let binary_path_in_final = format!("/home/appuser/{}", self.name); |                 let binary_path_in_final = format!("/home/appuser/{}", self.name); | ||||||
|                 dockerfile.push( |                 dockerfile.push( | ||||||
| @ -328,9 +327,8 @@ impl RustWebapp { | |||||||
|                 )); |                 )); | ||||||
| 
 | 
 | ||||||
|                 // Copy only the compiled binary from the builder stage.
 |                 // Copy only the compiled binary from the builder stage.
 | ||||||
|                 error!( |                 // TODO: Should not be using score name here, instead should use name from Cargo.toml
 | ||||||
|                     "FIXME Should not be using score name here, instead should use name from Cargo.toml" |                 // https://git.nationtech.io/NationTech/harmony/issues/105
 | ||||||
|                 ); |  | ||||||
|                 let binary_path_in_builder = format!("/app/target/release/{}", self.name); |                 let binary_path_in_builder = format!("/app/target/release/{}", self.name); | ||||||
|                 let binary_path_in_final = format!("/usr/local/bin/{}", self.name); |                 let binary_path_in_final = format!("/usr/local/bin/{}", self.name); | ||||||
|                 dockerfile.push( |                 dockerfile.push( | ||||||
|  | |||||||
| @ -1,9 +1,28 @@ | |||||||
| use non_blank_string_rs::NonBlankString; | use non_blank_string_rs::NonBlankString; | ||||||
| use std::str::FromStr; | use std::str::FromStr; | ||||||
| 
 | 
 | ||||||
| use crate::modules::helm::chart::{HelmChartScore, HelmRepository}; | use crate::{ | ||||||
|  |     modules::helm::chart::{HelmChartScore, HelmRepository}, | ||||||
|  |     topology::DeploymentTarget, | ||||||
|  | }; | ||||||
|  | 
 | ||||||
|  | pub fn ntfy_helm_chart_score( | ||||||
|  |     namespace: String, | ||||||
|  |     host: String, | ||||||
|  |     target: DeploymentTarget, | ||||||
|  | ) -> HelmChartScore { | ||||||
|  |     // TODO not actually the correct logic, this should be fixed by using an ingresss which is the
 | ||||||
|  |     // correct k8s standard.
 | ||||||
|  |     //
 | ||||||
|  |     // Another option is to delegate to the topology the ingress technology it wants to use Route,
 | ||||||
|  |     // Ingress or other
 | ||||||
|  |     let route_enabled = match target { | ||||||
|  |         DeploymentTarget::LocalDev => false, | ||||||
|  |         DeploymentTarget::Staging => true, | ||||||
|  |         DeploymentTarget::Production => true, | ||||||
|  |     }; | ||||||
|  |     let ingress_enabled = !route_enabled; | ||||||
| 
 | 
 | ||||||
| pub fn ntfy_helm_chart_score(namespace: String, host: String) -> HelmChartScore { |  | ||||||
|     let values = format!( |     let values = format!( | ||||||
|         r#" |         r#" | ||||||
| replicaCount: 1 | replicaCount: 1 | ||||||
| @ -25,23 +44,14 @@ serviceAccount: | |||||||
| 
 | 
 | ||||||
| service: | service: | ||||||
|   type: ClusterIP |   type: ClusterIP | ||||||
|   port: 80 |   port: 8080 | ||||||
| 
 | 
 | ||||||
| ingress: | ingress: | ||||||
|   enabled: true |   enabled: {ingress_enabled} | ||||||
| #  annotations: |  | ||||||
|     # kubernetes.io/ingress.class: nginx |  | ||||||
|     # kubernetes.io/tls-acme: "true" |  | ||||||
|   hosts: |  | ||||||
|     - host: {host} |  | ||||||
|       paths: |  | ||||||
|         - path: / |  | ||||||
|           pathType: ImplementationSpecific |  | ||||||
|   tls: [] |  | ||||||
|   #  - secretName: chart-example-tls |  | ||||||
|   #    hosts: |  | ||||||
|   #      - chart-example.local |  | ||||||
| 
 | 
 | ||||||
|  | route: | ||||||
|  |   enabled: {route_enabled} | ||||||
|  |   host: {host} | ||||||
| 
 | 
 | ||||||
| autoscaling: | autoscaling: | ||||||
|   enabled: false |   enabled: false | ||||||
| @ -49,7 +59,7 @@ autoscaling: | |||||||
| config: | config: | ||||||
|   enabled: true |   enabled: true | ||||||
|   data: |   data: | ||||||
| #    base-url: "https://ntfy.something.com" |     base-url: "https://{host}" | ||||||
|     auth-file: "/var/cache/ntfy/user.db" |     auth-file: "/var/cache/ntfy/user.db" | ||||||
|     auth-default-access: "deny-all" |     auth-default-access: "deny-all" | ||||||
|     cache-file: "/var/cache/ntfy/cache.db" |     cache-file: "/var/cache/ntfy/cache.db" | ||||||
| @ -59,6 +69,7 @@ config: | |||||||
|     enable-signup: false |     enable-signup: false | ||||||
|     enable-login: "true" |     enable-login: "true" | ||||||
|     enable-metrics: "true" |     enable-metrics: "true" | ||||||
|  |     listen-http: ":8080" | ||||||
| 
 | 
 | ||||||
| persistence: | persistence: | ||||||
|   enabled: true |   enabled: true | ||||||
| @ -69,16 +80,12 @@ persistence: | |||||||
|     HelmChartScore { |     HelmChartScore { | ||||||
|         namespace: Some(NonBlankString::from_str(&namespace).unwrap()), |         namespace: Some(NonBlankString::from_str(&namespace).unwrap()), | ||||||
|         release_name: NonBlankString::from_str("ntfy").unwrap(), |         release_name: NonBlankString::from_str("ntfy").unwrap(), | ||||||
|         chart_name: NonBlankString::from_str("sarab97/ntfy").unwrap(), |         chart_name: NonBlankString::from_str("oci://hub.nationtech.io/harmony/ntfy").unwrap(), | ||||||
|         chart_version: Some(NonBlankString::from_str("0.1.7").unwrap()), |         chart_version: Some(NonBlankString::from_str("0.1.7-nationtech.1").unwrap()), | ||||||
|         values_overrides: None, |         values_overrides: None, | ||||||
|         values_yaml: Some(values.to_string()), |         values_yaml: Some(values.to_string()), | ||||||
|         create_namespace: true, |         create_namespace: true, | ||||||
|         install_only: false, |         install_only: false, | ||||||
|         repository: Some(HelmRepository::new( |         repository: None, | ||||||
|             "sarab97".to_string(), |  | ||||||
|             url::Url::parse("https://charts.sarabsingh.com").unwrap(), |  | ||||||
|             true, |  | ||||||
|         )), |  | ||||||
|     } |     } | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,7 +1,7 @@ | |||||||
| use std::sync::Arc; | use std::sync::Arc; | ||||||
| 
 | 
 | ||||||
| use async_trait::async_trait; | use async_trait::async_trait; | ||||||
| use log::debug; | use log::info; | ||||||
| use serde::Serialize; | use serde::Serialize; | ||||||
| use strum::{Display, EnumString}; | use strum::{Display, EnumString}; | ||||||
| 
 | 
 | ||||||
| @ -11,7 +11,7 @@ use crate::{ | |||||||
|     inventory::Inventory, |     inventory::Inventory, | ||||||
|     modules::monitoring::ntfy::helm::ntfy_helm_chart::ntfy_helm_chart_score, |     modules::monitoring::ntfy::helm::ntfy_helm_chart::ntfy_helm_chart_score, | ||||||
|     score::Score, |     score::Score, | ||||||
|     topology::{HelmCommand, K8sclient, Topology, k8s::K8sClient}, |     topology::{HelmCommand, K8sclient, MultiTargetTopology, Topology, k8s::K8sClient}, | ||||||
| }; | }; | ||||||
| 
 | 
 | ||||||
| #[derive(Debug, Clone, Serialize)] | #[derive(Debug, Clone, Serialize)] | ||||||
| @ -20,7 +20,7 @@ pub struct NtfyScore { | |||||||
|     pub host: String, |     pub host: String, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| impl<T: Topology + HelmCommand + K8sclient> Score<T> for NtfyScore { | impl<T: Topology + HelmCommand + K8sclient + MultiTargetTopology> Score<T> for NtfyScore { | ||||||
|     fn create_interpret(&self) -> Box<dyn Interpret<T>> { |     fn create_interpret(&self) -> Box<dyn Interpret<T>> { | ||||||
|         Box::new(NtfyInterpret { |         Box::new(NtfyInterpret { | ||||||
|             score: self.clone(), |             score: self.clone(), | ||||||
| @ -77,7 +77,7 @@ impl NtfyInterpret { | |||||||
|                 vec![ |                 vec![ | ||||||
|                     "sh", |                     "sh", | ||||||
|                     "-c", |                     "-c", | ||||||
|                     format!("NTFY_PASSWORD={password} ntfy user add --role={role} {username}") |                     format!("NTFY_PASSWORD={password} ntfy user add --role={role} --ignore-exists {username}") | ||||||
|                         .as_str(), |                         .as_str(), | ||||||
|                 ], |                 ], | ||||||
|             ) |             ) | ||||||
| @ -89,22 +89,27 @@ impl NtfyInterpret { | |||||||
| 
 | 
 | ||||||
| /// We need a ntfy interpret to wrap the HelmChartScore in order to run the score, and then bootstrap the config inside ntfy
 | /// We need a ntfy interpret to wrap the HelmChartScore in order to run the score, and then bootstrap the config inside ntfy
 | ||||||
| #[async_trait] | #[async_trait] | ||||||
| impl<T: Topology + HelmCommand + K8sclient> Interpret<T> for NtfyInterpret { | impl<T: Topology + HelmCommand + K8sclient + MultiTargetTopology> Interpret<T> for NtfyInterpret { | ||||||
|     async fn execute( |     async fn execute( | ||||||
|         &self, |         &self, | ||||||
|         inventory: &Inventory, |         inventory: &Inventory, | ||||||
|         topology: &T, |         topology: &T, | ||||||
|     ) -> Result<Outcome, InterpretError> { |     ) -> Result<Outcome, InterpretError> { | ||||||
|         ntfy_helm_chart_score(self.score.namespace.clone(), self.score.host.clone()) |         ntfy_helm_chart_score( | ||||||
|  |             self.score.namespace.clone(), | ||||||
|  |             self.score.host.clone(), | ||||||
|  |             topology.current_target(), | ||||||
|  |         ) | ||||||
|         .interpret(inventory, topology) |         .interpret(inventory, topology) | ||||||
|         .await?; |         .await?; | ||||||
| 
 | 
 | ||||||
|         debug!("installed ntfy helm chart"); |         info!("installed ntfy helm chart"); | ||||||
|         let client = topology |         let client = topology | ||||||
|             .k8s_client() |             .k8s_client() | ||||||
|             .await |             .await | ||||||
|             .expect("couldn't get k8s client"); |             .expect("couldn't get k8s client"); | ||||||
| 
 | 
 | ||||||
|  |         info!("deploying ntfy..."); | ||||||
|         client |         client | ||||||
|             .wait_until_deployment_ready( |             .wait_until_deployment_ready( | ||||||
|                 "ntfy".to_string(), |                 "ntfy".to_string(), | ||||||
| @ -112,12 +117,12 @@ impl<T: Topology + HelmCommand + K8sclient> Interpret<T> for NtfyInterpret { | |||||||
|                 None, |                 None, | ||||||
|             ) |             ) | ||||||
|             .await?; |             .await?; | ||||||
|         debug!("created k8s client"); |         info!("ntfy deployed"); | ||||||
| 
 | 
 | ||||||
|  |         info!("adding user harmony"); | ||||||
|         self.add_user(client, "harmony", "harmony", Some(NtfyRole::Admin)) |         self.add_user(client, "harmony", "harmony", Some(NtfyRole::Admin)) | ||||||
|             .await?; |             .await?; | ||||||
| 
 |         info!("user added"); | ||||||
|         debug!("exec into pod done"); |  | ||||||
| 
 | 
 | ||||||
|         Ok(Outcome::success("Ntfy installed".to_string())) |         Ok(Outcome::success("Ntfy installed".to_string())) | ||||||
|     } |     } | ||||||
|  | |||||||
| @ -166,7 +166,8 @@ impl K8sPrometheusCRDAlertingInterpret { | |||||||
| 
 | 
 | ||||||
|         let install_output = Command::new("helm") |         let install_output = Command::new("helm") | ||||||
|             .args([ |             .args([ | ||||||
|                 "install", |                 "upgrade", | ||||||
|  |                 "--install", | ||||||
|                 &chart_name, |                 &chart_name, | ||||||
|                 tgz_path.to_str().unwrap(), |                 tgz_path.to_str().unwrap(), | ||||||
|                 "--namespace", |                 "--namespace", | ||||||
|  | |||||||
| @ -1,10 +1,16 @@ | |||||||
| use harmony::{ | use harmony::{ | ||||||
|     instrumentation::{self, HarmonyEvent}, |     instrumentation::{self, HarmonyEvent}, | ||||||
|  |     modules::application::ApplicationFeatureStatus, | ||||||
|     topology::TopologyStatus, |     topology::TopologyStatus, | ||||||
| }; | }; | ||||||
| use indicatif::MultiProgress; | use indicatif::MultiProgress; | ||||||
| use indicatif_log_bridge::LogWrapper; | use indicatif_log_bridge::LogWrapper; | ||||||
| use std::sync::{Arc, Mutex}; | use log::error; | ||||||
|  | use std::{ | ||||||
|  |     sync::{Arc, Mutex}, | ||||||
|  |     thread, | ||||||
|  |     time::Duration, | ||||||
|  | }; | ||||||
| 
 | 
 | ||||||
| use crate::progress::{IndicatifProgressTracker, ProgressTracker}; | use crate::progress::{IndicatifProgressTracker, ProgressTracker}; | ||||||
| 
 | 
 | ||||||
| @ -58,6 +64,7 @@ async fn handle_events(base_progress: MultiProgress) { | |||||||
|                             &format!("\n{} Harmony completed\n\n", crate::theme::EMOJI_HARMONY), |                             &format!("\n{} Harmony completed\n\n", crate::theme::EMOJI_HARMONY), | ||||||
|                         ); |                         ); | ||||||
|                         progress_tracker.add_section("harmony-finished", "\n\n"); |                         progress_tracker.add_section("harmony-finished", "\n\n"); | ||||||
|  |                         thread::sleep(Duration::from_millis(200)); | ||||||
|                         return false; |                         return false; | ||||||
|                     } |                     } | ||||||
|                     HarmonyEvent::TopologyStateChanged { |                     HarmonyEvent::TopologyStateChanged { | ||||||
| @ -156,10 +163,40 @@ async fn handle_events(base_progress: MultiProgress) { | |||||||
|                                 _ => progress_tracker.fail_task(&task_key, &outcome.message), |                                 _ => progress_tracker.fail_task(&task_key, &outcome.message), | ||||||
|                             }, |                             }, | ||||||
|                             Err(err) => { |                             Err(err) => { | ||||||
|  |                                 error!("Interpret error: {err}"); | ||||||
|                                 progress_tracker.fail_task(&task_key, &err.to_string()); |                                 progress_tracker.fail_task(&task_key, &err.to_string()); | ||||||
|                             } |                             } | ||||||
|                         } |                         } | ||||||
|                     } |                     } | ||||||
|  |                     HarmonyEvent::ApplicationFeatureStateChanged { | ||||||
|  |                         topology: _, | ||||||
|  |                         application, | ||||||
|  |                         feature, | ||||||
|  |                         status, | ||||||
|  |                     } => { | ||||||
|  |                         if let Some(score) = &(*current_score) { | ||||||
|  |                             let section_key = score_key(score); | ||||||
|  |                             let task_key = app_feature_key(&application, &feature); | ||||||
|  | 
 | ||||||
|  |                             match status { | ||||||
|  |                                 ApplicationFeatureStatus::Installing => { | ||||||
|  |                                     let message = format!("Feature '{}' installing...", feature); | ||||||
|  |                                     progress_tracker.add_task(§ion_key, &task_key, &message); | ||||||
|  |                                 } | ||||||
|  |                                 ApplicationFeatureStatus::Installed => { | ||||||
|  |                                     let message = format!("Feature '{}' installed", feature); | ||||||
|  |                                     progress_tracker.finish_task(&task_key, &message); | ||||||
|  |                                 } | ||||||
|  |                                 ApplicationFeatureStatus::Failed { details } => { | ||||||
|  |                                     let message = format!( | ||||||
|  |                                         "Feature '{}' installation failed: {}", | ||||||
|  |                                         feature, details | ||||||
|  |                                     ); | ||||||
|  |                                     progress_tracker.fail_task(&task_key, &message); | ||||||
|  |                                 } | ||||||
|  |                             } | ||||||
|  |                         } | ||||||
|  |                     } | ||||||
|                 } |                 } | ||||||
|                 true |                 true | ||||||
|             } |             } | ||||||
| @ -175,3 +212,7 @@ fn topology_key(topology: &str) -> String { | |||||||
| fn score_key(score: &str) -> String { | fn score_key(score: &str) -> String { | ||||||
|     format!("score-{score}") |     format!("score-{score}") | ||||||
| } | } | ||||||
|  | 
 | ||||||
|  | fn app_feature_key(application: &str, feature: &str) -> String { | ||||||
|  |     format!("app-{application}-{feature}") | ||||||
|  | } | ||||||
|  | |||||||
| @ -132,8 +132,9 @@ async fn init<T: Topology + Send + Sync + 'static>( | |||||||
| 
 | 
 | ||||||
|     // if list option is specified, print filtered list and exit
 |     // if list option is specified, print filtered list and exit
 | ||||||
|     if args.list { |     if args.list { | ||||||
|         println!("Available scores:"); |         let num_scores = scores_vec.len(); | ||||||
|         println!("{}", list_scores_with_index(&scores_vec)); |         println!("Available scores {num_scores}:"); | ||||||
|  |         println!("{}\n\n", list_scores_with_index(&scores_vec)); | ||||||
|         return Ok(()); |         return Ok(()); | ||||||
|     } |     } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -33,29 +33,13 @@ pub struct IndicatifProgressTracker { | |||||||
| 
 | 
 | ||||||
| impl IndicatifProgressTracker { | impl IndicatifProgressTracker { | ||||||
|     pub fn new(base: MultiProgress) -> Self { |     pub fn new(base: MultiProgress) -> Self { | ||||||
|         // The indicatif log bridge will insert a progress bar at the top.
 |         let sections = HashMap::new(); | ||||||
|         // To prevent our first section from being erased, we need to create
 |         let tasks = HashMap::new(); | ||||||
|         // 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 { |         let state = Arc::new(Mutex::new(IndicatifProgressTrackerState { | ||||||
|             sections, |             sections, | ||||||
|             tasks, |             tasks, | ||||||
|             pb_count: 1, |             pb_count: 0, | ||||||
|         })); |         })); | ||||||
| 
 | 
 | ||||||
|         Self { mp: base, state } |         Self { mp: base, state } | ||||||
|  | |||||||
| @ -21,10 +21,14 @@ lazy_static! { | |||||||
|     pub static ref SUCCESS_SPINNER_STYLE: ProgressStyle = SPINNER_STYLE |     pub static ref SUCCESS_SPINNER_STYLE: ProgressStyle = SPINNER_STYLE | ||||||
|         .clone() |         .clone() | ||||||
|         .tick_strings(&[format!("{}", EMOJI_SUCCESS).as_str()]); |         .tick_strings(&[format!("{}", EMOJI_SUCCESS).as_str()]); | ||||||
|     pub static ref SKIP_SPINNER_STYLE: ProgressStyle = SPINNER_STYLE |     pub static ref SKIP_SPINNER_STYLE: ProgressStyle = ProgressStyle::default_spinner() | ||||||
|  |         .template("    {spinner:.orange} {wide_msg}") | ||||||
|  |         .unwrap() | ||||||
|         .clone() |         .clone() | ||||||
|         .tick_strings(&[format!("{}", EMOJI_SKIP).as_str()]); |         .tick_strings(&[format!("{}", EMOJI_SKIP).as_str()]); | ||||||
|     pub static ref ERROR_SPINNER_STYLE: ProgressStyle = SPINNER_STYLE |     pub static ref ERROR_SPINNER_STYLE: ProgressStyle = ProgressStyle::default_spinner() | ||||||
|  |         .template("    {spinner:.red} {wide_msg}") | ||||||
|  |         .unwrap() | ||||||
|         .clone() |         .clone() | ||||||
|         .tick_strings(&[format!("{}", EMOJI_ERROR).as_str()]); |         .tick_strings(&[format!("{}", EMOJI_ERROR).as_str()]); | ||||||
| } | } | ||||||
|  | |||||||
| @ -1,6 +1,5 @@ | |||||||
| use harmony_cli::progress::{IndicatifProgressTracker, ProgressTracker}; | use harmony_cli::progress::{IndicatifProgressTracker, ProgressTracker}; | ||||||
| use indicatif::MultiProgress; | use indicatif::MultiProgress; | ||||||
| use log::error; |  | ||||||
| use std::sync::Arc; | use std::sync::Arc; | ||||||
| 
 | 
 | ||||||
| use crate::instrumentation::{self, HarmonyComposerEvent}; | use crate::instrumentation::{self, HarmonyComposerEvent}; | ||||||
| @ -53,15 +52,13 @@ pub async fn handle_events() { | |||||||
|                         progress_tracker.finish_task(COMPILTATION_TASK, "project compiled"); |                         progress_tracker.finish_task(COMPILTATION_TASK, "project compiled"); | ||||||
|                     } |                     } | ||||||
|                     HarmonyComposerEvent::ProjectCompilationFailed { details } => { |                     HarmonyComposerEvent::ProjectCompilationFailed { details } => { | ||||||
|                         progress_tracker.fail_task(COMPILTATION_TASK, "failed to compile project"); |                         progress_tracker.fail_task(COMPILTATION_TASK, &format!("failed to compile project:\n{details}")); | ||||||
| 
 |  | ||||||
|                         error!("{details}"); |  | ||||||
|                     } |                     } | ||||||
|                     HarmonyComposerEvent::DeploymentStarted { target } => { |                     HarmonyComposerEvent::DeploymentStarted { target, profile } => { | ||||||
|                         progress_tracker.add_section( |                         progress_tracker.add_section( | ||||||
|                             PROGRESS_DEPLOYMENT, |                             PROGRESS_DEPLOYMENT, | ||||||
|                             &format!( |                             &format!( | ||||||
|                                 "\n{} Deploying project to {target}...\n", |                                 "\n{} Deploying project on target '{target}' with profile '{profile}'...\n", | ||||||
|                                 harmony_cli::theme::EMOJI_DEPLOY, |                                 harmony_cli::theme::EMOJI_DEPLOY, | ||||||
|                             ), |                             ), | ||||||
|                         ); |                         ); | ||||||
| @ -69,6 +66,10 @@ pub async fn handle_events() { | |||||||
|                     HarmonyComposerEvent::DeploymentCompleted => { |                     HarmonyComposerEvent::DeploymentCompleted => { | ||||||
|                         progress_tracker.clear(); |                         progress_tracker.clear(); | ||||||
|                     } |                     } | ||||||
|  |                     HarmonyComposerEvent::DeploymentFailed { details } => { | ||||||
|  |                         progress_tracker.add_task(PROGRESS_DEPLOYMENT, "deployment-failed", ""); | ||||||
|  |                         progress_tracker.fail_task("deployment-failed", &details); | ||||||
|  |                     }, | ||||||
|                     HarmonyComposerEvent::Shutdown => { |                     HarmonyComposerEvent::Shutdown => { | ||||||
|                         return false; |                         return false; | ||||||
|                     } |                     } | ||||||
|  | |||||||
| @ -2,16 +2,28 @@ use log::debug; | |||||||
| use once_cell::sync::Lazy; | use once_cell::sync::Lazy; | ||||||
| use tokio::sync::broadcast; | use tokio::sync::broadcast; | ||||||
| 
 | 
 | ||||||
|  | use crate::{HarmonyProfile, HarmonyTarget}; | ||||||
|  | 
 | ||||||
| #[derive(Debug, Clone)] | #[derive(Debug, Clone)] | ||||||
| pub enum HarmonyComposerEvent { | pub enum HarmonyComposerEvent { | ||||||
|     HarmonyComposerStarted, |     HarmonyComposerStarted, | ||||||
|     ProjectInitializationStarted, |     ProjectInitializationStarted, | ||||||
|     ProjectInitialized, |     ProjectInitialized, | ||||||
|     ProjectCompilationStarted { details: String }, |     ProjectCompilationStarted { | ||||||
|  |         details: String, | ||||||
|  |     }, | ||||||
|     ProjectCompiled, |     ProjectCompiled, | ||||||
|     ProjectCompilationFailed { details: String }, |     ProjectCompilationFailed { | ||||||
|     DeploymentStarted { target: String }, |         details: String, | ||||||
|  |     }, | ||||||
|  |     DeploymentStarted { | ||||||
|  |         target: HarmonyTarget, | ||||||
|  |         profile: HarmonyProfile, | ||||||
|  |     }, | ||||||
|     DeploymentCompleted, |     DeploymentCompleted, | ||||||
|  |     DeploymentFailed { | ||||||
|  |         details: String, | ||||||
|  |     }, | ||||||
|     Shutdown, |     Shutdown, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | |||||||
| @ -49,14 +49,11 @@ struct CheckArgs { | |||||||
| 
 | 
 | ||||||
| #[derive(Args, Clone, Debug)] | #[derive(Args, Clone, Debug)] | ||||||
| struct DeployArgs { | struct DeployArgs { | ||||||
|     #[arg(long, default_value_t = false)] |     #[arg(long = "target", short = 't', default_value = "local")] | ||||||
|     staging: bool, |     harmony_target: HarmonyTarget, | ||||||
| 
 | 
 | ||||||
|     #[arg(long, default_value_t = false)] |     #[arg(long = "profile", short = 'p', default_value = "dev")] | ||||||
|     prod: bool, |     harmony_profile: HarmonyProfile, | ||||||
| 
 |  | ||||||
|     #[arg(long, default_value_t = false)] |  | ||||||
|     smoke_test: bool, |  | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| #[derive(Args, Clone, Debug)] | #[derive(Args, Clone, Debug)] | ||||||
| @ -68,6 +65,38 @@ struct AllArgs { | |||||||
|     deploy: DeployArgs, |     deploy: DeployArgs, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
|  | #[derive(Clone, Debug, clap::ValueEnum)] | ||||||
|  | enum HarmonyTarget { | ||||||
|  |     Local, | ||||||
|  |     Remote, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl std::fmt::Display for HarmonyTarget { | ||||||
|  |     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||||
|  |         match self { | ||||||
|  |             HarmonyTarget::Local => f.write_str("local"), | ||||||
|  |             HarmonyTarget::Remote => f.write_str("remote"), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | #[derive(Clone, Debug, clap::ValueEnum)] | ||||||
|  | enum HarmonyProfile { | ||||||
|  |     Dev, | ||||||
|  |     Staging, | ||||||
|  |     Production, | ||||||
|  | } | ||||||
|  | 
 | ||||||
|  | impl std::fmt::Display for HarmonyProfile { | ||||||
|  |     fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result { | ||||||
|  |         match self { | ||||||
|  |             HarmonyProfile::Dev => f.write_str("dev"), | ||||||
|  |             HarmonyProfile::Staging => f.write_str("staging"), | ||||||
|  |             HarmonyProfile::Production => f.write_str("production"), | ||||||
|  |         } | ||||||
|  |     } | ||||||
|  | } | ||||||
|  | 
 | ||||||
| #[tokio::main] | #[tokio::main] | ||||||
| async fn main() { | async fn main() { | ||||||
|     let hc_logger_handle = harmony_composer_logger::init(); |     let hc_logger_handle = harmony_composer_logger::init(); | ||||||
| @ -122,26 +151,39 @@ async fn main() { | |||||||
|                 ); |                 ); | ||||||
|             } |             } | ||||||
|             Commands::Deploy(args) => { |             Commands::Deploy(args) => { | ||||||
|                 let deploy = if args.staging { |  | ||||||
|                 instrumentation::instrument(HarmonyComposerEvent::DeploymentStarted { |                 instrumentation::instrument(HarmonyComposerEvent::DeploymentStarted { | ||||||
|                         target: "staging".to_string(), |                     target: args.harmony_target.clone(), | ||||||
|  |                     profile: args.harmony_profile.clone(), | ||||||
|                 }) |                 }) | ||||||
|                 .unwrap(); |                 .unwrap(); | ||||||
|                     todo!("implement staging deployment") | 
 | ||||||
|                 } else if args.prod { |                 if matches!(args.harmony_profile, HarmonyProfile::Dev) | ||||||
|                     instrumentation::instrument(HarmonyComposerEvent::DeploymentStarted { |                     && !matches!(args.harmony_target, HarmonyTarget::Local) | ||||||
|                         target: "prod".to_string(), |                 { | ||||||
|                     }) |                     instrumentation::instrument(HarmonyComposerEvent::DeploymentFailed { | ||||||
|                     .unwrap(); |                         details: format!( | ||||||
|                     todo!("implement prod deployment") |                             "Cannot run profile '{}' on target '{}'. Profile '{}' can run locally only.", | ||||||
|                 } else { |                             args.harmony_profile, args.harmony_target, args.harmony_profile | ||||||
|                     instrumentation::instrument(HarmonyComposerEvent::DeploymentStarted { |                         ), | ||||||
|                         target: "dev".to_string(), |                     }).unwrap(); | ||||||
|                     }) |                     return; | ||||||
|                     .unwrap(); |  | ||||||
|                     Command::new(harmony_bin_path).arg("-y").arg("-a").spawn() |  | ||||||
|                 } |                 } | ||||||
|                 .expect("failed to run harmony deploy"); | 
 | ||||||
|  |                 let use_local_k3d = match args.harmony_target { | ||||||
|  |                     HarmonyTarget::Local => true, | ||||||
|  |                     HarmonyTarget::Remote => false, | ||||||
|  |                 }; | ||||||
|  | 
 | ||||||
|  |                 let mut command = Command::new(harmony_bin_path); | ||||||
|  |                 command | ||||||
|  |                     .env("HARMONY_USE_LOCAL_K3D", format!("{use_local_k3d}")) | ||||||
|  |                     .env("HARMONY_PROFILE", format!("{}", args.harmony_profile)) | ||||||
|  |                     .arg("-y") | ||||||
|  |                     .arg("-a"); | ||||||
|  | 
 | ||||||
|  |                 info!("{:?}", command); | ||||||
|  | 
 | ||||||
|  |                 let deploy = command.spawn().expect("failed to run harmony deploy"); | ||||||
| 
 | 
 | ||||||
|                 let deploy_output = deploy.wait_with_output().unwrap(); |                 let deploy_output = deploy.wait_with_output().unwrap(); | ||||||
|                 debug!("{}", String::from_utf8(deploy_output.stdout).unwrap()); |                 debug!("{}", String::from_utf8(deploy_output.stdout).unwrap()); | ||||||
|  | |||||||
		Loading…
	
		Reference in New Issue
	
	Block a user