forked from NationTech/harmony
		
	
		
			
				
	
	
		
			162 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
			
		
		
	
	
			162 lines
		
	
	
		
			4.4 KiB
		
	
	
	
		
			Rust
		
	
	
	
	
	
| use async_trait::async_trait;
 | |
| use email_address::EmailAddress;
 | |
| 
 | |
| use log::info;
 | |
| use serde::Serialize;
 | |
| use url::Url;
 | |
| 
 | |
| use crate::{
 | |
|     data::{Id, Version},
 | |
|     interpret::{Interpret, InterpretError, InterpretName, InterpretStatus, Outcome},
 | |
|     inventory::Inventory,
 | |
|     score::Score,
 | |
|     topology::{HelmCommand, Topology},
 | |
| };
 | |
| 
 | |
| use super::{
 | |
|     config::KubePrometheusConfig, discord_alert_manager::discord_alert_manager_score,
 | |
|     kube_prometheus::kube_prometheus_helm_chart_score,
 | |
| };
 | |
| 
 | |
| #[derive(Debug, Clone, Serialize)]
 | |
| pub enum AlertChannel {
 | |
|     Discord {
 | |
|         name: String,
 | |
|         webhook_url: Url,
 | |
|     },
 | |
|     Slack {
 | |
|         slack_channel: String,
 | |
|         webhook_url: Url,
 | |
|     },
 | |
|     //TODO test and implement in helm chart
 | |
|     //currently does not work
 | |
|     Smpt {
 | |
|         email_address: EmailAddress,
 | |
|         service_name: String,
 | |
|     },
 | |
| }
 | |
| 
 | |
| #[derive(Debug, Clone, Serialize)]
 | |
| pub struct MonitoringAlertingStackScore {
 | |
|     pub alert_channel: Vec<AlertChannel>,
 | |
|     pub namespace: Option<String>,
 | |
| }
 | |
| 
 | |
| impl MonitoringAlertingStackScore {
 | |
|     pub fn new() -> Self {
 | |
|         Self {
 | |
|             alert_channel: Vec::new(),
 | |
|             namespace: None,
 | |
|         }
 | |
|     }
 | |
| }
 | |
| 
 | |
| impl<T: Topology + HelmCommand> Score<T> for MonitoringAlertingStackScore {
 | |
|     fn create_interpret(&self) -> Box<dyn Interpret<T>> {
 | |
|         Box::new(MonitoringAlertingStackInterpret {
 | |
|             score: self.clone(),
 | |
|         })
 | |
|     }
 | |
|     fn name(&self) -> String {
 | |
|         format!("MonitoringAlertingStackScore")
 | |
|     }
 | |
| }
 | |
| 
 | |
| #[derive(Debug, Clone, Serialize)]
 | |
| struct MonitoringAlertingStackInterpret {
 | |
|     score: MonitoringAlertingStackScore,
 | |
| }
 | |
| 
 | |
| impl MonitoringAlertingStackInterpret {
 | |
|     async fn build_kube_prometheus_helm_chart_config(&self) -> KubePrometheusConfig {
 | |
|         let mut config = KubePrometheusConfig::new();
 | |
|         if let Some(ns) = &self.score.namespace {
 | |
|             config.namespace = ns.clone();
 | |
|         }
 | |
|         config.alert_channel = self.score.alert_channel.clone();
 | |
|         config
 | |
|     }
 | |
| 
 | |
|     async fn deploy_kube_prometheus_helm_chart_score<T: Topology + HelmCommand>(
 | |
|         &self,
 | |
|         inventory: &Inventory,
 | |
|         topology: &T,
 | |
|         config: &KubePrometheusConfig,
 | |
|     ) -> Result<Outcome, InterpretError> {
 | |
|         let helm_chart = kube_prometheus_helm_chart_score(config);
 | |
|         helm_chart
 | |
|             .create_interpret()
 | |
|             .execute(inventory, topology)
 | |
|             .await
 | |
|     }
 | |
| 
 | |
|     async fn deploy_alert_channel_service<T: Topology + HelmCommand>(
 | |
|         &self,
 | |
|         inventory: &Inventory,
 | |
|         topology: &T,
 | |
|         config: &KubePrometheusConfig,
 | |
|     ) -> Result<Outcome, InterpretError> {
 | |
|         let mut outcomes = vec![];
 | |
| 
 | |
|         for channel in &self.score.alert_channel {
 | |
|             let outcome = match channel {
 | |
|                 AlertChannel::Discord { .. } => {
 | |
|                     discord_alert_manager_score(config)
 | |
|                         .create_interpret()
 | |
|                         .execute(inventory, topology)
 | |
|                         .await
 | |
|                 }
 | |
|                 AlertChannel::Slack { .. } => Ok(Outcome::success(
 | |
|                     "No extra configs for slack alerting".to_string(),
 | |
|                 )),
 | |
|                 AlertChannel::Smpt { .. } => {
 | |
|                     todo!()
 | |
|                 }
 | |
|             };
 | |
|             outcomes.push(outcome);
 | |
|         }
 | |
|         for result in outcomes {
 | |
|             result?; 
 | |
|         }
 | |
| 
 | |
|         Ok(Outcome::success("All alert channels deployed".to_string()))
 | |
|     }
 | |
| }
 | |
| 
 | |
| #[async_trait]
 | |
| impl<T: Topology + HelmCommand> Interpret<T> for MonitoringAlertingStackInterpret {
 | |
|     async fn execute(
 | |
|         &self,
 | |
|         inventory: &Inventory,
 | |
|         topology: &T,
 | |
|     ) -> Result<Outcome, InterpretError> {
 | |
|         let config = self.build_kube_prometheus_helm_chart_config().await;
 | |
|         info!("Built kube prometheus config");
 | |
|         info!("Installing kube prometheus chart");
 | |
|         self.deploy_kube_prometheus_helm_chart_score(inventory, topology, &config)
 | |
|             .await?;
 | |
|         info!("Installing alert channel service");
 | |
|         self.deploy_alert_channel_service(inventory, topology, &config)
 | |
|             .await?;
 | |
|         Ok(Outcome::success(format!(
 | |
|             "succesfully deployed monitoring and alerting stack"
 | |
|         )))
 | |
|     }
 | |
| 
 | |
|     fn get_name(&self) -> InterpretName {
 | |
|         todo!()
 | |
|     }
 | |
| 
 | |
|     fn get_version(&self) -> Version {
 | |
|         todo!()
 | |
|     }
 | |
| 
 | |
|     fn get_status(&self) -> InterpretStatus {
 | |
|         todo!()
 | |
|     }
 | |
| 
 | |
|     fn get_children(&self) -> Vec<Id> {
 | |
|         todo!()
 | |
|     }
 | |
| }
 |