Here you will find an example plugin that covers the basics for writing a stream collector plugin.
For your streaming collector plugin, create a new repository and name your plugin project using the following format:
snap-plugin-[plugin-type]-[plugin-name]
For example:
snap-plugin-collector-rand
Example files and directory structure:
snap-plugin-[plugin-type]-[plugin-name]
|--[plugin-name]
|--[plugin-name].go
|--[plugin-name]_test.go
|--main.go
For example:
snap-plugin-collector-rand
|--rand
|--rand.go
|--rand_test.go
|--main.go
- The [plugin-name] folder (for example
rand
) will include all files to implement the appropriate interface methods - Your [plugin-name] folder will also include your test files.
In order to write a plugin for Snap, it is necessary to define a few methods to satisfy the appropriate interfaces. These interfaces must be defined for a streaming collector plugin:
// Plugin is an interface shared by all plugins and must implemented by each plugin to communicate with Snap.
type Plugin interface {
GetConfigPolicy() (ConfigPolicy, error)
}
// StreamCollector is a Collector that can send back metrics on its own
// defined interval (within configurable limits). These limits are set by the
// SetMaxBuffer and SetMaxCollectionDuration funcs.
type StreamCollector interface {
Plugin
// StreamMetrics allows the plugin to send/receive metrics on a channel
// Arguments are (in order):
//
// A channel for metrics into the plugin from Snap -- which
// are the metric types snap is requesting the plugin to collect.
//
// A channel for metrics from the plugin to Snap -- the actual
// collected metrics from the plugin.
//
// A channel for error strings that the library will report to snap
// as task errors.
StreamMetrics(chan []Metric, chan []Metric, chan string) error
// SetMaxbuffer sets the maximum number of metrics the plugin should buffer
// before sending metrics.
SetMaxBuffer(int64)
// SetMaxCollectionDuration sets the maximum duration between collections
// before metrics should be sent (i.e.5s MaxCollectionDuration means that after
// 5 seconds, the plugin should send whatever it has instead of waiting longer).
SetMaxCollectDuration(time.Duration)
GetMetricTypes(Config) ([]Metric, error)
}
The interface is slightly different depending on what type (collector, processor, or publisher, streaming collector) of plugin is being written. Please see other plugin types for more details.
After implementing a type that satisfies one of {collector, processor, publisher, streaming collector} interfaces, all that is left to do is to call the appropriate plugin.StartX() with your plugin specific meta options. For example, with no meta options specified:
plugin.StartStreamCollector(&rand.RandCollector{}, pluginName, pluginVersion)
The available options are defined in plugin/meta.go. You can use some or none of the options. The options with definitions/explanations are below:
// ConcurrencyCount is the max number of concurrent calls the plugin
// should take. For example:
// If there are 5 tasks using the plugin and its concurrency count is 2,
// snapd will keep 3 plugin instances running.
// ConcurrencyCount overwrites the default (5) for a Meta's ConcurrencyCount.
func ConcurrencyCount(cc int) MetaOpt {
}
// Exclusive == true results in a single instance of the plugin running
// regardless of the number of tasks using the plugin.
// Exclusive overwrites the default (false) for a Meta's Exclusive key.
func Exclusive(e bool) MetaOpt {
}
// Unsecure results in unencrypted communication with this plugin.
// Unsecure overwrites the default (false) for a Meta's Unsecure key.
func Unsecure(e bool) MetaOpt {
}
// RoutingStrategy will override the routing strategy this plugin requires.
// The default routing strategy is Least Recently Used.
// RoutingStrategy overwrites the default (LRU) for a Meta's RoutingStrategy.
func RoutingStrategy(r router) MetaOpt {
}
// CacheTTL will override the default cache TTL for this plugin. snapd
// caches metrics on the daemon side for a default of 500ms.
// CacheTTL overwrites the default (500ms) for a Meta's CacheTTL.
func CacheTTL(t time.Duration) MetaOpt {
}
An example using some arbitrary values:
plugin.StartCollector(
mypackage.Mytype{},
pluginName,
pluginVersion,
plugin.ConcurrencyCount(2),
plugin.Exclusive(true),
plugin.Unsecure(true),
plugin.RoutingStrategy(StickyRouter),
plugin.CacheTTL(time.Second))
For testing reference the Snap Testing Guidelines. To test your plugin with Snap you will need to have Snap installed, check out these docs for Snap setup details.
Each test file should specify the appropriate build tag such as "small", "medium", "large" (e.g. // +build small).
For example, if you want to run only small tests:
// +build small
// you must include at least one line between the build tag and the package name.
package rand
For example, if you want to run small and medium tests:
// +build small medium
package rand
You've made a plugin! Now it's time to share it. Create a release by following these steps. We recommend that your release version match your plugin version, see example here.
Don't forget to announce your plugin release on slack and get your plugin added to the Plugin Catalog!