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.6.1"
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:
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
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] bindgen = ["collectd-plugin/bindgen"] default = [] ```
COLLECTD_VERSION
as 5.4
, 5.5
, or 5.7
, or rely on collectd-rust-plugin
auto detecting the version by executing collectd -h
.collectd-dev
lib
, so cp target/debug/libmyplugin.so /usr/lib/collectd/myplugin.so
LoadPlugin myplugin
to collectd.confThe load plugin in examples/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
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.