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.8.2"

Serde support is enabled by default for configuration parsing.

Then put this in your crate root:

rust extern crate collectd_plugin;

This repo is tested on the following:

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

[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(&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, 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 snap), you'll be fine.

Plugins

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