feat/monitoring_alerting #61

Merged
johnride merged 12 commits from feat/monitoring_alerting into master 2025-06-19 14:37:20 +00:00
22 changed files with 170 additions and 656 deletions
Showing only changes of commit 1771f47eca - Show all commits

View File

@ -4,10 +4,7 @@ use harmony::{
maestro::Maestro,
modules::{
lamp::{LAMPConfig, LAMPScore},
monitoring::{
alert_channel::discord_alert_channel::DiscordWebhook,
monitoring_alerting::MonitoringAlertingScore,
},
monitoring::alert_channel::discord_alert_channel::DiscordWebhook,
},
topology::{K8sAnywhereTopology, Url},
};
@ -35,15 +32,15 @@ async fn main() {
},
};
let monitoring = MonitoringAlertingScore {
alert_receivers: vec![Box::new(DiscordWebhook {
url: Url::Url(url::Url::parse("https://discord.idonotexist.com").unwrap()),
// TODO write url macro
// url: url!("https://discord.idonotexist.com"),
})],
alert_rules: vec![],
scrape_targets: vec![],
};
//let monitoring = MonitoringAlertingScore {
// alert_receivers: vec![Box::new(DiscordWebhook {
// url: Url::Url(url::Url::parse("https://discord.idonotexist.com").unwrap()),
// // TODO write url macro
// // url: url!("https://discord.idonotexist.com"),
// })],
// alert_rules: vec![],
// scrape_targets: vec![],
//};
// You can choose the type of Topology you want, we suggest starting with the
// K8sAnywhereTopology as it is the most automatic one that enables you to easily deploy
@ -57,7 +54,6 @@ async fn main() {
.unwrap();
// maestro.register_all(vec![Box::new(lamp_stack)]);
maestro.register_all(vec![Box::new(monitoring)]);
// Here we bootstrap the CLI, this gives some nice features if you need them
harmony_cli::init(maestro, None).await.unwrap();
}

View File

@ -1,6 +1,7 @@
use harmony::{
inventory::Inventory, maestro::Maestro,
modules::monitoring::alert_score::HelmPrometheusAlertingScore, topology::K8sAnywhereTopology,
modules::monitoring::kube_prometheus::helm_prometheus_alert_score::HelmPrometheusAlertingScore,
topology::K8sAnywhereTopology,
};
#[tokio::main]

View File

@ -0,0 +1,8 @@
use async_trait::async_trait;
use crate::interpret::InterpretError;
#[async_trait]
pub trait Installable {
async fn ensure_installed(&self) -> Result<(), InterpretError>;
}

View File

@ -1,6 +1,7 @@
mod ha_cluster;
mod host_binding;
mod http;
pub mod installable;
mod k8s_anywhere;
mod localhost;
pub mod oberservability;

View File

@ -1,71 +0,0 @@
use std::sync::Arc;
use async_trait::async_trait;
use serde::Serialize;
use crate::score::Score;
use crate::topology::HelmCommand;
use crate::{
interpret::{InterpretError, Outcome},
inventory::Inventory,
topology::Topology,
};
use super::{
K8sMonitorConfig,
monitoring::{AlertChannel, AlertChannelConfig, Monitor},
};
#[derive(Debug, Clone, Serialize)]
pub struct K8sMonitor {
pub config: K8sMonitorConfig,
}
#[async_trait]
impl Monitor for K8sMonitor {
async fn provision_monitor<T: Topology + HelmCommand>(
&self,
inventory: &Inventory,
topology: &T,
alert_channels: Option<Vec<Box<dyn AlertChannelConfig>>>,
) -> Result<Outcome, InterpretError> {
if let Some(channels) = alert_channels {
let alert_channels = self.build_alert_channels(channels).await?;
for channel in alert_channels {
channel.register_alert_channel().await?;
}
}
let chart = self.config.chart.clone();
chart
.create_interpret()
.execute(inventory, topology)
.await?;
Ok(Outcome::success("installed monitor".to_string()))
}
fn delete_monitor(&self) -> Result<Outcome, InterpretError> {
todo!()
}
}
#[async_trait]
impl AlertChannelConfig for K8sMonitor {
async fn build_alert_channel(&self) -> Result<Box<dyn AlertChannel>, InterpretError> {
todo!()
}
}
impl K8sMonitor {
pub async fn build_alert_channels(
&self,
alert_channel_configs: Vec<Box<dyn AlertChannelConfig>>,
) -> Result<Vec<Box<dyn AlertChannel>>, InterpretError> {
let mut alert_channels = Vec::new();
for config in alert_channel_configs {
let channel = config.build_alert_channel().await?;
alert_channels.push(channel)
}
Ok(alert_channels)
}
}

View File

@ -1,23 +1 @@
use serde::Serialize;
use crate::modules::{
helm::chart::HelmChartScore,
monitoring::kube_prometheus::kube_prometheus_helm_chart_score::kube_prometheus_helm_chart_score,
};
pub mod k8s;
pub mod monitoring;
#[derive(Debug, Clone, Serialize)]
pub struct K8sMonitorConfig {
//probably need to do something better here
pub chart: HelmChartScore,
}
impl K8sMonitorConfig {
pub fn cluster_monitor() -> Self {
Self {
chart: kube_prometheus_helm_chart_score(),
}
}
}

View File

@ -1,39 +1,61 @@
use async_trait::async_trait;
use dyn_clone::DynClone;
use std::fmt::Debug;
use crate::executors::ExecutorError;
use crate::interpret::InterpretError;
use crate::{
data::{Id, Version},
interpret::{Interpret, InterpretError, InterpretName, InterpretStatus, Outcome},
inventory::Inventory,
topology::{Topology, installable::Installable},
};
use crate::inventory::Inventory;
use crate::topology::HelmCommand;
use crate::{interpret::Outcome, topology::Topology};
pub trait AlertSender: Send + Sync + std::fmt::Debug + Installable {}
#[derive(Debug)]
pub struct AlertingInterpret<S: AlertSender> {
pub sender: S,
pub receivers: Vec<Box<dyn AlertReceiver<S>>>,
}
/// Represents an entity responsible for collecting and organizing observability data
/// from various telemetry sources such as Prometheus or Datadog
/// 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 {
async fn provision_monitor<T: Topology + HelmCommand>(
impl<S: AlertSender, T: Topology> Interpret<T> for AlertingInterpret<S> {
async fn execute(
&self,
inventory: &Inventory,
topology: &T,
alert_receivers: Option<Vec<Box<dyn AlertChannelConfig>>>,
) -> Result<Outcome, InterpretError>;
) -> Result<Outcome, InterpretError> {
for receiver in self.receivers.iter() {
receiver.install(&self.sender).await?;
}
todo!()
}
fn delete_monitor(&self) -> Result<Outcome, InterpretError>;
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!()
}
}
#[async_trait]
pub trait AlertChannel: Debug + Send + Sync {
async fn register_alert_channel(&self) -> Result<Outcome, ExecutorError>;
//async fn get_channel_id(&self) -> String;
pub trait AlertReceiver<S: AlertSender>: std::fmt::Debug + Send + Sync {
async fn install(&self, sender: &S) -> Result<(), InterpretError>;
}
#[async_trait]
pub trait AlertChannelConfig: Debug + Send + Sync + DynClone {
async fn build_alert_channel(&self) -> Result<Box<dyn AlertChannel>, InterpretError>;
pub trait AlertRule<S: AlertSender> {
async fn install(&self, sender: &S) -> Result<(), InterpretError>;
}
dyn_clone::clone_trait_object!(AlertChannelConfig);
#[async_trait]
pub trait ScrapeTarger<S: AlertSender> {
async fn install(&self, sender: &S) -> Result<(), InterpretError>;
}

View File

@ -1,54 +1,20 @@
use log::debug;
use serde::Serialize;
use async_trait::async_trait;
use crate::{
modules::monitoring::{
alert_receiver::AlertReceiver,
prometheus::{Prometheus, PrometheusReceiver},
},
topology::Url,
interpret::InterpretError,
modules::monitoring::kube_prometheus::prometheus::{Prometheus, PrometheusReceiver},
topology::{Url, oberservability::monitoring::AlertReceiver},
};
#[derive(Debug, Clone, Serialize)]
#[derive(Debug)]
pub struct DiscordWebhook {
pub name: String,
pub url: Url,
}
impl AlertReceiver for DiscordWebhook {
type M = Prometheus;
fn install(&self, sender: &Self::M) -> Result<(), String> {
sender.configure_receiver(Box::new(self))?;
debug!("DiscordWebhook installed for Prometheus");
Ok(())
}
fn clone_box(&self) -> Box<dyn AlertReceiver<M = Self::M>> {
Box::new(self.clone())
#[async_trait]
impl AlertReceiver<Prometheus> for DiscordWebhook {
async fn install(&self, sender: &Prometheus) -> Result<(), InterpretError> {
sender.install_receiver(PrometheusReceiver {}).await
}
}
impl PrometheusReceiver for DiscordWebhook {
fn get_prometheus_config(
&self,
) -> crate::modules::monitoring::prometheus::PrometheusReceiverConfig {
todo!()
}
}
// #[derive(Debug, Clone, Serialize)]
// pub struct DiscordWebhookScore {
// pub config: DiscordWebhook,
// }
//
// impl<T: Topology> Score<T> for DiscordWebhookScore {
// fn create_interpret(&self) -> Box<dyn Interpret<T>> {
// Box::new(AlertReceiverInterpret {
// receiver: Box::new(self.config.clone()),
// })
// }
//
// fn name(&self) -> String {
// todo!()
// }
// }

View File

@ -1,57 +0,0 @@
use std::fmt::Debug;
use async_trait::async_trait;
use crate::{
data::{Id, Version},
interpret::{Interpret, InterpretError, InterpretName, InterpretStatus, Outcome},
inventory::Inventory,
topology::Topology,
};
use super::monitoring_alerting::MonitoringSystem;
pub trait AlertReceiver: Debug + Send + Sync {
type M: MonitoringSystem;
fn install(&self, sender: &Self::M) -> Result<(), String>;
fn clone_box(&self) -> Box<dyn AlertReceiver<M = Self::M>>;
}
struct AlertReceiverConfig<M: MonitoringSystem> {
config: String, // Or whatever
sender: M,
}
#[derive(Debug)]
pub struct AlertReceiverInterpret<M: MonitoringSystem> {
pub receiver: Box<dyn AlertReceiver<M = M>>,
}
#[async_trait]
impl<T: Topology, M: MonitoringSystem> Interpret<T> for AlertReceiverInterpret<M> {
async fn execute(
&self,
_inventory: &Inventory,
_topology: &T,
) -> Result<Outcome, InterpretError> {
todo!()
}
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!()
}
}

View File

@ -1,13 +0,0 @@
use std::fmt::Debug;
use dyn_clone::DynClone;
use super::monitoring_alerting::MonitoringSystem;
pub trait AlertRule: Debug + Send + Sync + DynClone {
type M: MonitoringSystem;
fn install(&self, sender: &Self::M);
fn clone_box(&self) -> Box<dyn AlertRule<M = Self::M>>;
}

View File

@ -1,148 +0,0 @@
use async_trait::async_trait;
use serde::{Serializer, Serialize};
use crate::{
data::{Id, Version},
interpret::{Interpret, InterpretError, InterpretName, InterpretStatus, Outcome},
inventory::Inventory,
score::Score,
topology::{HelmCommand, Topology, Url},
};
#[async_trait]
pub trait Installable {
async fn ensure_installed(&self) -> Result<(), InterpretError>;
}
pub trait AlertSender: Send + Sync + std::fmt::Debug + Installable {}
#[async_trait]
pub trait AlertReceiver<S: AlertSender>: std::fmt::Debug + Send + Sync {
async fn install(&self, sender: &S) -> Result<(), InterpretError>;
}
#[derive(Debug)]
pub struct DiscordWebhook {
name: String,
url: Url,
}
#[async_trait]
impl AlertReceiver<Prometheus> for DiscordWebhook {
async fn install(&self, sender: &Prometheus) -> Result<(), InterpretError> {
sender
.install_receiver(PrometheusReceiver {
})
.await
}
}
#[derive(Clone, Debug, Serialize)]
pub struct HelmPrometheusAlertingScore {
pub receivers: Vec<Box<dyn AlertReceiver<Prometheus>>>,
}
impl Serialize for Box<dyn AlertReceiver<Prometheus>> {
fn serialize<S>(&self, _serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
todo!()
}
}
impl Clone for Box<dyn AlertReceiver<Prometheus>> {
fn clone(&self) -> Self {
todo!()
}
}
impl<T: Topology + HelmCommand> Score<T> for HelmPrometheusAlertingScore {
fn create_interpret(&self) -> Box<dyn crate::interpret::Interpret<T>> {
Box::new(AlertingInterpret {
sender: Prometheus {},
receivers: vec![Box::new(DiscordWebhook {url:todo!(), name: todo!() })],
})
}
fn name(&self) -> String {
"HelmPrometheusAlertingScore".to_string()
}
}
#[derive(Debug)]
pub struct Prometheus;
impl Prometheus {
async fn install_receiver(
&self,
prometheus_receiver: PrometheusReceiver,
) -> Result<(), InterpretError> {
todo!()
}
}
pub struct PrometheusReceiver {
}
impl PrometheusReceiver {
fn get_prometheus_receiver_config(&self) {}
}
pub struct AlertChannelGlobalConfig {}
#[derive(Serialize)]
pub struct AlertReceiverRoute {
pub receiver: String,
pub matchers: Vec<String>,
#[serde(default)]
pub r#continue: bool,
}
pub struct AlertChannelReceiver {
pub name: String,
}
impl AlertSender for Prometheus {}
#[async_trait]
impl Installable for Prometheus {
async fn ensure_installed(&self) -> Result<(), InterpretError> {
todo!()
}
}
#[derive(Debug)]
pub struct AlertingInterpret<S: AlertSender> {
pub sender: S,
pub receivers: Vec<Box<dyn AlertReceiver<S>>>,
}
#[async_trait]
impl<S: AlertSender, T: Topology> Interpret<T> for AlertingInterpret<S> {
#[must_use]
async fn execute(
&self,
inventory: &Inventory,
topology: &T,
) -> Result<Outcome, InterpretError> {
for receiver in self.receivers.iter() {
receiver.install(&self.sender).await?;
}
todo!()
}
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!()
}
}

View File

@ -1,7 +1,5 @@
use serde::Serialize;
use super::types::AlertManagerChannelConfig;
#[derive(Debug, Clone, Serialize)]
pub struct KubePrometheusConfig {
pub namespace: String,
@ -21,7 +19,6 @@ pub struct KubePrometheusConfig {
pub kube_proxy: bool,
pub kube_state_metrics: bool,
pub prometheus_operator: bool,
pub alert_channels: Vec<AlertManagerChannelConfig>,
}
impl KubePrometheusConfig {
pub fn new() -> Self {
@ -43,7 +40,6 @@ impl KubePrometheusConfig {
prometheus_operator: true,
core_dns: false,
kube_scheduler: false,
alert_channels: Vec::new(),
}
}
}

View File

@ -0,0 +1,2 @@
pub mod config;
pub mod kube_prometheus_helm_chart;

View File

@ -0,0 +1,47 @@
use serde::Serialize;
use crate::{
modules::monitoring::alert_channel::discord_alert_channel::DiscordWebhook,
score::Score,
topology::{
HelmCommand, Topology,
oberservability::monitoring::{AlertReceiver, AlertingInterpret},
},
};
use super::prometheus::Prometheus;
#[derive(Clone, Debug, Serialize)]
pub struct HelmPrometheusAlertingScore {
pub receivers: Vec<Box<dyn AlertReceiver<Prometheus>>>,
}
impl<T: Topology + HelmCommand> Score<T> for HelmPrometheusAlertingScore {
fn create_interpret(&self) -> Box<dyn crate::interpret::Interpret<T>> {
Box::new(AlertingInterpret {
sender: Prometheus {},
receivers: vec![Box::new(DiscordWebhook {
url: todo!(),
name: todo!(),
})],
})
}
fn name(&self) -> String {
"HelmPrometheusAlertingScore".to_string()
}
}
impl Serialize for Box<dyn AlertReceiver<Prometheus>> {
fn serialize<S>(&self, _serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
todo!()
}
}
impl Clone for Box<dyn AlertReceiver<Prometheus>> {
fn clone(&self) -> Self {
todo!()
}
}

View File

@ -1,3 +1,4 @@
pub mod config;
pub mod kube_prometheus_helm_chart_score;
pub mod helm;
pub mod helm_prometheus_alert_score;
pub mod prometheus;
pub mod types;

View File

@ -0,0 +1,34 @@
use async_trait::async_trait;
use crate::{
interpret::InterpretError,
topology::{installable::Installable, oberservability::monitoring::AlertSender},
};
impl AlertSender for Prometheus {}
#[async_trait]
impl Installable for Prometheus {
async fn ensure_installed(&self) -> Result<(), InterpretError> {
todo!()
}
}
#[derive(Debug)]
pub struct Prometheus;
impl Prometheus {
pub async fn install_receiver(
&self,
prometheus_receiver: PrometheusReceiver,
) -> Result<(), InterpretError> {
todo!()
}
}
pub struct PrometheusReceiver {}
impl PrometheusReceiver {
fn get_prometheus_receiver_config(&self) {}
}
pub struct AlertChannelGlobalConfig {}

View File

@ -1,14 +1,12 @@
use serde::Serialize;
#[derive(Debug, Clone, Serialize)]
pub struct AlertManagerChannelConfig {
pub global_configs: AlertManagerChannelGlobalConfigs,
pub route: AlertManagerChannelRoute,
pub receiver: AlertManagerChannelReceiver,
#[derive(Serialize)]
pub struct AlertReceiverRoute {
pub receiver: String,
pub matchers: Vec<String>,
#[serde(default)]
pub r#continue: bool,
}
pub struct AlertChannelReceiver {
pub name: String,
}
#[derive(Debug, Clone, Serialize)]
pub struct AlertManagerChannelGlobalConfigs {}
#[derive(Debug, Clone, Serialize)]
pub struct AlertManagerChannelReceiver {}
#[derive(Debug, Clone, Serialize)]
pub struct AlertManagerChannelRoute {}

View File

@ -1,8 +1,2 @@
pub mod alert_channel;
pub mod alert_receiver;
pub mod alert_rule;
pub mod kube_prometheus;
pub mod monitoring_alerting;
pub mod prometheus;
pub mod scrape_target;
pub mod alert_score;

View File

@ -1,115 +0,0 @@
use std::sync::Arc;
use async_trait::async_trait;
use log::debug;
use serde::Serialize;
use crate::{
data::{Id, Version},
interpret::{Interpret, InterpretError, InterpretName, InterpretStatus, Outcome},
inventory::Inventory,
score::Score,
topology::Topology,
};
use super::{alert_receiver::AlertReceiver, prometheus::Installable};
use super::alert_rule::AlertRule;
use super::scrape_target::ScrapeTarget;
#[async_trait]
pub trait MonitoringSystem: std::fmt::Debug + Clone + Serialize + 'static + Send + Sync {
}
#[derive(Debug, Clone)]
pub struct MonitoringAlertingScore<M: MonitoringSystem> {
pub alert_receivers: Vec<Box<dyn AlertReceiver<M = M>>>,
pub alert_rules: Vec<Box<dyn AlertRule<M = M>>>,
pub scrape_targets: Vec<Box<dyn ScrapeTarget<M = M>>>,
}
impl<M: MonitoringSystem> Clone for Box<dyn AlertReceiver<M = M>> {
fn clone(&self) -> Self {
self.clone_box()
}
}
impl<M: MonitoringSystem> Clone for Box<dyn AlertRule<M = M>> {
fn clone(&self) -> Self {
self.clone_box()
}
}
impl<M: MonitoringSystem> Clone for Box<dyn ScrapeTarget<M = M>> {
fn clone(&self) -> Self {
self.clone_box()
}
}
impl<M: MonitoringSystem> Serialize for MonitoringAlertingScore<M> {
fn serialize<S>(&self, _serializer: S) -> Result<S::Ok, S::Error>
where
S: serde::Serializer,
{
todo!()
}
}
impl<T: Topology, M: MonitoringSystem + Serialize + Installable> Score<T>
for MonitoringAlertingScore<M>
{
fn create_interpret(&self) -> Box<dyn Interpret<T>> {
Box::new(MonitoringAlertingInterpret {
score: Arc::new(self.clone()),
})
}
fn name(&self) -> String {
"MonitoringAlertingScore".to_string()
}
}
#[derive(Debug)]
struct MonitoringAlertingInterpret<M: MonitoringSystem> {
score: Arc<MonitoringAlertingScore<M>>,
}
#[async_trait]
impl<M: MonitoringSystem + Installable, T: Topology> Interpret<T>
for MonitoringAlertingInterpret<M>
{
async fn execute(
&self,
_inventory: &Inventory,
_topology: &T,
) -> Result<Outcome, InterpretError> {
debug!("score {:#?}", self.score);
todo!("Figure out a clean way to have things come together here :
We want Prometheus to be bound like this :
DiscordWebhook (AlertReceiver<Sender = Prometheus>)
|
Prometheus (AlertSender + Scraper + Installable<T: HelmCommand>)
^
MonitoringScore {{ alert_rules, alert_receivers, scrape_endpoints }}
(Interpret<T: Topology + Installer
")
}
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!()
}
}

View File

@ -1,112 +0,0 @@
use async_trait::async_trait;
use serde::Serialize;
use serde_value::Value;
use crate::{
data::{Id, Version},
interpret::{Interpret, InterpretError, InterpretName, InterpretStatus, Outcome},
inventory::Inventory,
topology::{K8sAnywhereTopology, Topology},
};
use std::fmt::Debug;
use super::monitoring_alerting::MonitoringSystem;
#[derive(Debug, Clone, Serialize)]
pub struct Prometheus {}
impl Prometheus {
pub fn configure_receiver(&self, receiver: Box<&dyn PrometheusReceiver>) -> Result<(), String> {
let _receiver_config = receiver.get_prometheus_config();
todo!()
}
pub fn configure_rule(&self, _rule: Box<&dyn PrometheusRule>) {
todo!()
}
pub fn configure_scrape_target(&self, _target: Box<&dyn PrometheusScrapeTarget>) {
todo!()
}
}
impl MonitoringSystem for Prometheus {}
pub trait PrometheusCapability {
fn install_alert_receivers(&self, receivers: Vec<Box<dyn PrometheusReceiver>>);
fn install_alert_rules(&self, rules: Vec<Box<dyn PrometheusRule>>);
fn install_scrape_targets(&self, receivers: Vec<Box<dyn PrometheusScrapeTarget>>);
}
pub trait PrometheusReceiver {
fn get_prometheus_config(&self) -> PrometheusReceiverConfig;
}
pub struct PrometheusReceiverConfig {
config: Value, // either a serde Value or a more specific type that understands prometheus
// config
}
pub trait PrometheusRule {
fn get_prometheus_config(&self) -> PrometheusRuleConfig;
}
pub struct PrometheusRuleConfig {
pub definition: String, // Not a string but an actual prometheus rule config
}
pub trait PrometheusScrapeTarget {
fn get_prometheus_config(&self) -> PrometheusScrapeTargetConfig;
}
pub struct PrometheusScrapeTargetConfig {
pub definition: String, // Not a string but an actual prometheus scraping config
}
pub struct PrometheusMonitoringScore {
alert_receivers: Vec<Box<PrometheusReceiverConfig>>,
alert_rules: Vec<Box<PrometheusRuleConfig>>,
scrape_targets: Vec<Box<PrometheusScrapeTargetConfig>>,
}
#[derive(Debug)]
pub struct PrometheusMonitoringInterpret {}
#[async_trait]
impl<T: Topology + PrometheusCapability> Interpret<T> for PrometheusMonitoringInterpret {
async fn execute(
&self,
_inventory: &Inventory,
_topology: &T,
) -> Result<Outcome, InterpretError> {
todo!()
}
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!()
}
}
#[async_trait]
pub trait Installable {
async fn install(&self) -> Result<Outcome, InterpretError>;
type Installer;
}
#[async_trait]
impl Installable for Prometheus {
type Installer = K8sAnywhereTopology;
async fn install(&self) -> Result<Outcome, InterpretError> {
todo!()
}
}

View File

@ -1,14 +0,0 @@
use std::fmt::Debug;
use dyn_clone::DynClone;
use super::monitoring_alerting::MonitoringSystem;
pub trait ScrapeTarget: Debug + Send + Sync + DynClone {
type M: MonitoringSystem;
fn install(&self, sender: &Self::M);
fn clone_box(&self) -> Box<dyn ScrapeTarget<M = Self::M>>;
}