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:
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:
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 plugins/load
```rust
extern crate collectd_plugin; extern crate failure;
use collectd_plugin::{ConfigItem, Plugin, PluginCapabilities, PluginManager, PluginRegistration, Value, ValueListBuilder}; use failure::Error;
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); ```
There are five main ways to extend collectd:
<collectd/core/daemon/plugin.h>
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 ensure a successful build, adapt the below to your project's Cargo file.
```toml
[lib]
crate-type = ["cdylib"]
name = "
[features] collectd-54 = ["collectd-plugin/collectd-54"] collectd-55 = ["collectd-plugin/collectd-55"] collectd-57 = ["collectd-plugin/collectd-57"] bindgen = ["collectd-plugin/bindgen"] default = [] ```
collectd-dev
lib
, so cp target/debug/libmyplugin.so /usr/lib/collectd/myplugin.so
LoadPlugin myplugin
to collectd.confThe load plugin in plugins/load demonstrates how to expose configuration values to Collectd.
```xml
To measure the overhead of adapting Collectd's datatypes when writing and reporting values:
bash
cargo bench --features 'stub collectd-57'
If you'd like to use the timings on my machine:
ValueListBuilder
ValueList
for plugins that write valuesUnless 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.