A Collectd Plugin Written 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.4.1"

If you want Serde support (recommended), include: features like this:

toml [dependencies.collectd-plugin] version = "0.4.1" features = ["serde"]

Then put this in your crate root:

rust extern crate collectd_plugin;

Rust 1.20 or later is needed to build.

This repo is tested on the following:

Quickstart

Below is a complete plugin that dummy reports load values to collectd, as it registers a READ hook. For an implementation has the same behavior as Collectd's own load plugin, see plugins/load

```rust

[macro_use]

extern crate collectd_plugin; extern crate failure;

use collectd_plugin::{ConfigItem, Plugin, PluginCapabilities, PluginManager, PluginRegistration, Value, ValueListBuilder}; use failure::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, 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(&mut self) -> Result<(), 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()
}

}

// 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, the following steps are needed:

Plugin Configuration

The load plugin in plugins/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 ```