From 0d56fbc09d6c46b9b2e8ff6f93fb4604325c6b54 Mon Sep 17 00:00:00 2001 From: Willem Date: Mon, 2 Jun 2025 14:44:43 -0400 Subject: [PATCH] wip: applied comments in pr, changed naming of AlertChannel to AlertReceiver and added rust doc to Monitor for clarity --- .../topology/oberservability/monitoring.rs | 22 ++++++---- .../monitoring/discord_alert_manager.rs | 31 ++++---------- .../monitoring/discord_webhook_sender.rs | 34 +++++++-------- .../modules/monitoring/monitoring_alerting.rs | 42 +++++++++---------- 4 files changed, 61 insertions(+), 68 deletions(-) diff --git a/harmony/src/domain/topology/oberservability/monitoring.rs b/harmony/src/domain/topology/oberservability/monitoring.rs index 485f633..35e8b33 100644 --- a/harmony/src/domain/topology/oberservability/monitoring.rs +++ b/harmony/src/domain/topology/oberservability/monitoring.rs @@ -8,26 +8,32 @@ use crate::interpret::InterpretError; use crate::{interpret::Outcome, topology::Topology}; +/// Represents an entity responsible for collecting and organizing observability data +/// from various telemetry sources +/// A `Monitor` abstracts the logic required to scrape, aggregate, and structure +/// monitoring data, enabling consistent processing regardless of the underlying data source. #[async_trait] pub trait Monitor: Debug + Send + Sync { async fn deploy_monitor( &self, topology: &T, - config: Vec>>, + alert_receivers: Vec>>, ) -> Result; async fn delete_monitor( &self, topolgy: &T, - config: Vec>>, + alert_receivers: Vec>>, ) -> Result; } #[async_trait] -pub trait AlertChannelConfig: Debug + DynClone + Send + Sync { - fn channel_identifier(&self) -> String; - - fn webhook_url(&self) -> Option; - - fn send_resolved_notifications(&self) -> bool; +pub trait AlertReceiver: Debug + DynClone + Send + Sync { + fn get_alert_receiver_config(&self) -> AlertReceiverConfig; +} + +pub struct AlertReceiverConfig { + pub receiver_id: String, + pub receiver_webhook_url: Option, + pub send_resolved: bool, } diff --git a/harmony/src/modules/monitoring/discord_alert_manager.rs b/harmony/src/modules/monitoring/discord_alert_manager.rs index a3519e8..7765505 100644 --- a/harmony/src/modules/monitoring/discord_alert_manager.rs +++ b/harmony/src/modules/monitoring/discord_alert_manager.rs @@ -1,30 +1,25 @@ use std::str::FromStr; use non_blank_string_rs::NonBlankString; +use url::Url; use crate::modules::helm::chart::HelmChartScore; -use super::{config::KubePrometheusConfig, monitoring_alerting::AlertChannel}; - -fn get_discord_alert_manager_score(config: &KubePrometheusConfig) -> Option { - let (url, name) = config.alert_channel.iter().find_map(|channel| { - if let AlertChannel::Discord { webhook_url, name } = channel { - Some((webhook_url, name)) - } else { - None - } - })?; - +pub fn discord_alert_manager_score( + webhook_url: Url, + namespace: String, + name: String, +) -> HelmChartScore { let values = format!( r#" environment: - name: "DISCORD_WEBHOOK" - value: "{url}" + value: "{webhook_url}" "#, ); - Some(HelmChartScore { - namespace: Some(NonBlankString::from_str(&config.namespace).unwrap()), + HelmChartScore { + namespace: Some(NonBlankString::from_str(&namespace).unwrap()), release_name: NonBlankString::from_str(&name).unwrap(), chart_name: NonBlankString::from_str( "oci://hub.nationtech.io/library/alertmanager-discord", @@ -36,13 +31,5 @@ environment: create_namespace: true, install_only: true, repository: None, - }) -} - -pub fn discord_alert_manager_score(config: &KubePrometheusConfig) -> HelmChartScore { - if let Some(chart) = get_discord_alert_manager_score(config) { - chart - } else { - panic!("Expected discord alert manager helm chart"); } } diff --git a/harmony/src/modules/monitoring/discord_webhook_sender.rs b/harmony/src/modules/monitoring/discord_webhook_sender.rs index e5eea07..28bdcb9 100644 --- a/harmony/src/modules/monitoring/discord_webhook_sender.rs +++ b/harmony/src/modules/monitoring/discord_webhook_sender.rs @@ -3,7 +3,10 @@ use url::Url; use crate::{ interpret::{InterpretError, Outcome}, - topology::{K8sAnywhereTopology, Topology, oberservability::monitoring::AlertChannelConfig}, + topology::{ + K8sAnywhereTopology, Topology, + oberservability::monitoring::{AlertReceiver, AlertReceiverConfig}, + }, }; #[derive(Debug, Clone)] @@ -13,41 +16,38 @@ pub struct DiscordWebhookConfig { pub send_resolved_notifications: bool, } -pub trait DiscordWebhookSender { - fn deploy_discord_webhook_sender( +pub trait DiscordWebhookReceiver { + fn deploy_discord_webhook_receiver( &self, _notification_adapter_id: &str, ) -> Result; - fn delete_discord_webhook_sender( + fn delete_discord_webhook_receiver( &self, _notification_adapter_id: &str, ) -> Result; } #[async_trait] -impl AlertChannelConfig for DiscordWebhookConfig { - fn channel_identifier(&self) -> String { - self.name.clone() - } - fn webhook_url(&self) -> Option { - Some(self.webhook_url.clone()) - } - - fn send_resolved_notifications(&self) -> bool { - self.send_resolved_notifications.clone() +impl AlertReceiver for DiscordWebhookConfig { + fn get_alert_receiver_config(&self) -> AlertReceiverConfig { + AlertReceiverConfig { + receiver_id: self.name.clone(), + receiver_webhook_url: Some(self.webhook_url.clone()), + send_resolved: self.send_resolved_notifications.clone(), + } } } #[async_trait] -impl DiscordWebhookSender for K8sAnywhereTopology { - fn deploy_discord_webhook_sender( +impl DiscordWebhookReceiver for K8sAnywhereTopology { + fn deploy_discord_webhook_receiver( &self, _notification_adapter_id: &str, ) -> Result { todo!() } - fn delete_discord_webhook_sender( + fn delete_discord_webhook_receiver( &self, _notification_adapter_id: &str, ) -> Result { diff --git a/harmony/src/modules/monitoring/monitoring_alerting.rs b/harmony/src/modules/monitoring/monitoring_alerting.rs index 325f9d3..6d2db38 100644 --- a/harmony/src/modules/monitoring/monitoring_alerting.rs +++ b/harmony/src/modules/monitoring/monitoring_alerting.rs @@ -96,28 +96,28 @@ impl MonitoringAlertingStackInterpret { topology: &T, config: &KubePrometheusConfig, ) -> Result { - let mut outcomes = vec![]; + //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?; - } + //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())) }