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.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,) |
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;
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); ```
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
= 5.4
, 5.5
, or 5.7
.COLLECTD_PATH
points to the root git directory for collectd. This option makes the most sense when coupled with the bindgen
feature.collectd -h
.collectd-dev
(if not using COLLECTD_PATH
)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 bottleneck), you'll be fine.
Do you use collectd-rust-plugin? Feel free to add your plugin to the list.