Build Status Rust Version

Write a Collectd Plugin in Rust

Collectd is a ubiquitous system statistics collection daemon. collectd_plugin leverages collectd's ability to dynamically load plugins and creates an ergonomic, yet extremely low cost abstraction API to interface with collectd.

Features:

Usage

Add to your Cargo.toml:

toml [dependencies] collectd-plugin = "0.12.0"

Serde support is enabled by default for configuration parsing.

Works with any collectd version 5.4+, but all users will need to specify the collectd api version they want to target via the COLLECTD_VERSION environment variable (or rely on $(collectd -h) or COLLECTD_PATH variable).

| COLLECTED_VERSION | Compatible Range | |---------------------|-------------------| | 5.4 | [5.4, 5.5) | | 5.5 | [5.5, 5.7) | | 5.7 | [5.7,) |

Quickstart

See what to add to your project's Cargo file

Below is a complete plugin that dummy reports load values to collectd, as it registers a READ hook. For an implementation that reimplements collectd's own load plugin, see examples/load

```rust use collectdplugin::{ collectdplugin, ConfigItem, Plugin, PluginCapabilities, PluginManager, PluginRegistration, Value, ValueListBuilder, }; use std::error;

[derive(Default)]

struct MyPlugin;

// A manager decides the name of the family of plugins and also registers one or more plugins based // on collectd's configuration files impl PluginManager for MyPlugin { // A plugin needs a unique name to be referenced by collectd fn name() -> &'static str { "myplugin" }

// Our plugin might have configuration section in collectd.conf, which will be passed here if
// present. Our contrived plugin doesn't care about configuration so it returns only a single
// plugin (itself).
fn plugins(
    _config: Option<&[ConfigItem<'_>]>,
) -> Result<PluginRegistration, Box<dyn error::Error>> {
    Ok(PluginRegistration::Single(Box::new(MyPlugin)))
}

}

impl Plugin for MyPlugin { // We define that our plugin will only be reporting / submitting values to writers fn capabilities(&self) -> PluginCapabilities { PluginCapabilities::READ }

fn read_values(&self) -> Result<(), Box<dyn error::Error>> {
    // Create a list of values to submit to collectd. We'll be sending in a vector representing the
    // "load" type. Short-term load is first (15.0) followed by mid-term and long-term. The number
    // of values that you submit at a time depends on types.db in collectd configurations
    let values = vec![Value::Gauge(15.0), Value::Gauge(10.0), Value::Gauge(12.0)];

    // Submit our values to collectd. A plugin can submit any number of times.
    ValueListBuilder::new(Self::name(), "load")
        .values(&values)
        .submit()?;

    Ok(())
}

}

// We pass in our plugin manager type collectd_plugin!(MyPlugin); ```

Motivation

There are five main ways to extend collectd:

And my thoughts:

Rust's combination of ecosystem, package manager, C ffi, single file dynamic library, and optimized code made it seem like a natural choice.

To Build

To ensure a successful build, adapt the below to your project's Cargo file.

```toml [lib] crate-type = ["cdylib"] name = ""

[features] bindgen = ["collectd-plugin/bindgen"] default = [] ```

Plugin Configuration

The load plugin in examples/load demonstrates how to expose configuration values to collectd.

```xml

In this example configuration we provide short and long term load and leave

Mid to the default value. Yes, this is very much contrived

ReportRelative true ```

Benchmarking Overhead

To measure the overhead of adapting collectd's datatypes when writing and reporting values:

bash cargo bench --features stub

If you'd like to use the timings on my machine:

Unless you are reporting or writing millions of metrics every interval (in which case you'll most likely hit an earlier bottleneck), you'll be fine.

Plugins

Do you use collectd-rust-plugin? Feel free to add your plugin to the list.