alumet 0.8.0

Modular framework for hardware and software measurement (including energy consumption and more).
Documentation
//! Definition of Rust plugins.
//!
//! See the [documentation of the plugin module](super#static-plugins).

use anyhow::{anyhow, Context};

use crate::plugin::{AlumetPluginStart, Plugin};

use super::{phases::AlumetPreStart, AlumetPostStart, ConfigTable};

/// Trait for Alumet plugins written in Rust.
///
/// Implement this trait to define your plugin.
/// See the [plugin module documentation](super#static-plugins).
pub trait AlumetPlugin {
    // Note: add `where Self: Sized` to make this trait "object safe", if necessary in the future.

    /// The name of the plugin. It must be unique: two plugins cannot have the same name.
    fn name() -> &'static str;

    /// The version of the plugin, for instance `"1.2.3"`. It should adhere to semantic versioning.
    fn version() -> &'static str;

    /// Initializes the plugin.
    ///
    /// Read more about the plugin lifecycle in the [module documentation](super).
    fn init(config: ConfigTable) -> anyhow::Result<Box<Self>>;

    /// Returns the default configuration of the plugin.
    ///
    /// # Example
    /// ```ignore
    /// use serde::{Deserialize, Serialize}
    /// use alumet::plugin::{
    ///     rust::{serialize_config, AlumetPlugin},
    ///     ConfigTable
    /// };
    ///
    /// #[derive(Deserialize, Serialize)]
    /// struct Config {
    ///     integer: i32
    /// }
    ///
    /// impl Default for Config {
    ///     fn default() -> Self { integer: 123 }
    /// }
    ///
    /// impl AlumetPlugin for MyPlugin {
    ///     fn default_config() -> anyhow::Result<Option<ConfigTable>> {
    ///         Ok(Some(serialize_config(Config::default())?))
    ///     }
    /// }
    /// ```
    fn default_config() -> anyhow::Result<Option<ConfigTable>>;

    /// Starts the plugin, allowing it to register metrics, sources and outputs.
    ///
    /// # Plugin restart
    /// A plugin can be started and stopped multiple times, for instance when ALUMET switches from monitoring to profiling mode.
    /// [`AlumetPlugin::stop`] is guaranteed to be called between two calls of [`AlumetPlugin::start`].
    fn start(&mut self, alumet: &mut AlumetPluginStart) -> anyhow::Result<()>;

    /// Stops the plugin.
    ///
    /// This method is called _after_ all the metrics, sources and outputs previously registered
    /// by [`AlumetPlugin::start`] have been stopped and unregistered.
    fn stop(&mut self) -> anyhow::Result<()>;

    /// Function called after the startup phase but before the operation phase,
    /// i.e. the measurement pipeline has not started yet.
    ///
    /// It can be used, for instance, to obtain the list of all registered metrics.
    fn pre_pipeline_start(&mut self, alumet: &mut AlumetPreStart) -> anyhow::Result<()> {
        let _ = alumet; // do nothing by default
        Ok(())
    }

    /// Function called after the beginning of the operation phase,
    /// i.e. the measurement pipeline has started.
    ///
    /// It can be used, for instance, to obtain a [`ScopedControlHandle`](crate::pipeline::control::ScopedControlHandle).
    /// No modification to the pipeline can be applied.
    fn post_pipeline_start(&mut self, alumet: &mut AlumetPostStart) -> anyhow::Result<()> {
        let _ = alumet; // do nothing by default
        Ok(())
    }
}

// Every AlumetPlugin is a Plugin :)
impl<P: AlumetPlugin> Plugin for P {
    fn name(&self) -> &str {
        P::name() as _
    }

    fn version(&self) -> &str {
        P::version() as _
    }

    fn start(&mut self, alumet: &mut AlumetPluginStart) -> anyhow::Result<()> {
        AlumetPlugin::start(self, alumet)
    }

    fn stop(&mut self) -> anyhow::Result<()> {
        AlumetPlugin::stop(self)
    }

    fn pre_pipeline_start(&mut self, alumet: &mut AlumetPreStart) -> anyhow::Result<()> {
        AlumetPlugin::pre_pipeline_start(self, alumet)
    }

    fn post_pipeline_start(&mut self, alumet: &mut AlumetPostStart) -> anyhow::Result<()> {
        AlumetPlugin::post_pipeline_start(self, alumet)
    }
}

pub fn deserialize_config<'de, T: serde::de::Deserialize<'de>>(config: ConfigTable) -> anyhow::Result<T> {
    toml::Value::Table(config.0)
        .try_into::<T>()
        .with_context(|| format!("error when deserializing ConfigTable to {}", std::any::type_name::<T>()))
        .context(InvalidConfig)
}

pub fn serialize_config<T: serde::ser::Serialize>(config: T) -> anyhow::Result<ConfigTable> {
    let res = match toml::Value::try_from(config) {
        Ok(toml::Value::Table(t)) => Ok(ConfigTable(t)),
        Ok(wrong) => Err(anyhow!(
            "{} did not get serialized to a toml Table but to a {}",
            std::any::type_name::<T>(),
            wrong.type_str()
        )),
        Err(e) => Err(anyhow!(
            "error when serializing {} to ConfigTable: {e}",
            std::any::type_name::<T>()
        )),
    };
    res.context(InvalidConfig)
}

/// Signals an invalid configuration.
///
/// Use this singleton with [`anyhow::Context`] to signal that
/// an error has been caused by an invalid configuration.
/// This will allow the Alumet application to display more helpful
/// messages to the user.
///
/// `InvalidConfig` is automatically applied by [`serialize_config`] and [`deserialize_config`] if an error occurs.
#[derive(Debug)]
pub struct InvalidConfig;

impl std::error::Error for InvalidConfig {}
impl std::fmt::Display for InvalidConfig {
    fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
        write!(f, "invalid configuration")
    }
}