From ef8a71f3c230b4afc0cf5c3f1936f61808448a27 Mon Sep 17 00:00:00 2001 From: Nan Liu Date: Thu, 10 Nov 2016 13:41:46 -0800 Subject: [PATCH] (SDI-2181) document new plugin authoring and publishing --- docs/PLUGIN_AUTHORING.md | 219 ++++++++++++++++------------------ docs/PLUGIN_BEST_PRACTICES.md | 23 ---- 2 files changed, 106 insertions(+), 136 deletions(-) delete mode 100644 docs/PLUGIN_BEST_PRACTICES.md diff --git a/docs/PLUGIN_AUTHORING.md b/docs/PLUGIN_AUTHORING.md index b262baaa2..4b4cd0734 100644 --- a/docs/PLUGIN_AUTHORING.md +++ b/docs/PLUGIN_AUTHORING.md @@ -17,152 +17,145 @@ See the License for the specific language governing permissions and limitations under the License. --> -## About This -The following is a recipe for authoring a plugin that fits smoothly within the snap framework. Like any recipe, the ingredients and the order in which you mix them are important. The major steps are: +# Plugin Authoring -1. Outline your plugin metrics -2. Decide the CODEC for the plugin -3. Download or clone [Snap](https://github.com/intelsdi-x/snap) -4. Download or clone [snap-plugin-utilities](https://github.com/intelsdi-x/snap-plugin-utilities) -5. Implement the required interfaces -6. Test the plugin -7. Expose the plugin +### Table of Content -Like any good recipe, it will do you well to read the entire document, as well as the [Plugin Best Practices](https://github.com/intelsdi-x/snap/blob/master/docs/PLUGIN_BEST_PRACTICES.md), before you start cooking. +1. [Overview](#overview) + * [Plugin Library](#plugin-library) +2. [Developing Plugins](#developing-plugins) + * [Plugin Type](#plugin-type) + * [Plugin Name](#plugin-name) + * [Plugin Metric Namespace](#plugin-metric-namespace) + * [Plugin Interface](#plugin-interface) + * [Plugin Version](#plugin-version) + * [Plugin Release](#plugin-release) + * [Plugin Metadata](#plugin-metadata) + * [Documentation](#documentation) -Bon Appétit! :stew: +## Overview -## Plugin Authoring -Snap itself runs as a master daemon with the core functionality that may load and unload plugin processes via either CLI or HTTP APIs. +Snap daemon runs as a service which provides the core functionality such as plugin management and task scheduling. Snap plugins extend Snap by providing additional capability to gather, transform, or publish metrics. Since plugins are standalone programs, they can be written in any language supported by [gRPC](http://grpc.io). -A Snap plugin is a program, or a set of functions or services, written in Go or any language; that may seamlessly integrate with snap as executables. +### Plugin Library -Communication between Snap and plugins uses RPC either through HTTP or TCP protocols. HTTP gRPC is good for any language to use that gRPC supports (see [gRPC docs](http://grpc.io/docs)) while the native client is only suitable for plugins written in Golang. When a plugin is written using one of the available snap-plugin-libs ([snap-plugin-lib-go](https://github.com/intelsdi-x/snap-plugin-lib-go), [snap-plugin-lib-py](https://github.com/intelsdi-x/snap-plugin-lib-py), or [snap-plugin-lib-cpp](https://github.com/intelsdi-x/snap-plugin-lib-cpp)) Google Protobuf, a binary serialization format, is used to encode/decode the data (see [plugin.proto](https://github.com/intelsdi-x/snap/blob/master/control/plugin/rpc/plugin.proto)). +The following libraries are available to simplify the process of writing a plugin: -Before starting writing Snap plugins, check out the [Plugin Catalog](https://github.com/intelsdi-x/snap/blob/master/docs/PLUGIN_CATALOG.md) to see if any suit your needs. If not, you need to reference the plugin packages that defines the type of structures and interfaces inside snap and then write plugin endpoints to implement the defined interfaces. +* [snap-plugin-lib-go](https://github.com/intelsdi-x/snap-plugin-lib-go) +* [snap-plugin-lib-cpp](https://github.com/intelsdi-x/snap-plugin-lib-cpp) +* [snap-plugin-lib-py](https://github.com/intelsdi-x/snap-plugin-lib-py) -### Plugin Naming, Files, and Directory -Snap supports three type of plugins. They are collectors, processors, and publishers. The plugin project name should use the following format: ->snap-plugin-[type]-[name] +A few notes before we get started: -For example: ->snap-plugin-collector-hana ->snap-plugin-processor-movingaverage ->snap-plugin-publisher-influxdb +Communication between Snap daemon and plugins use gRPC. So even if a plugin library isn't available in the language of your choice, you can still write a plugin using the [gRPC library](http://grpc.io/docs) as a starting point. However this requires additional knowledge about Snap API, [gRPC/protobuf](../control/plugin/rpc/plugin.proto), so it is beyond the scope of this document. -Example files and directory structure: -``` -snap-plugin-[type]-[name] - |--[name] - |--[name].go - |--[name]_test.go - |--[name]_integration_test.go - |--main.go - |--main_test.go -``` +Before writing a new Snap plugin, please check out the [Plugin Catalog](./PLUGIN_CATALOG.md) to see if any existing plugins meet your needs. If you need any assistance, please reach out on [Slack #snap-developers channel](https://intelsdi-x.herokuapp.com/). -### Metric Naming -A plugin should **NOT** advertise metrics which namespaces contain: +## Developing Plugins -##### a) the following characters in a namespace: - - spaces ` ` - - brackets: `()[]{}` - - slashes: `| \ /` - - carets: `^` - - quotations: `" ' \`` - - other punctuations: `. , ; ? !` +### Plugin Type -##### b) a wildcard in the end +Snap supports three type of plugins: -Example: +* collector: gathering metrics +* processor: transforming metrics +* publisher: publishing metrics -| Unacceptable metric namespace | Why | Proposal | -|:------------------------------|:-----------------------|:------------------------------------------| -| /intel/foo/\* | a wildcard in the end | /intel/foo/\*/bar
/intel/foo/\*/baz | -| /intel/mock/bar(no) | not allowed characters | /intel/mock/bar_no | -| /intel/mock/bar("no") | not allowed characters | /intel/mock/bar_no | -| /intel/mock/bar^no | not allowed characters | /intel/mock/bar_no | -| /intel/mock/bar.no | not allowed characters | /intel/mock/bar_no | -| /intel/mock/bar!? | not allowed characters | /intel/mock/bar | +### Plugin Name +The plugin repo name should follow this convention: `snap-plugin-[type]-[name]` -Snap validates the metrics exposed by the plugin and, if validation fails, an error is returned and the plugin is not loaded. +For example: +* `snap-plugin-collector-hana` +* `snap-plugin-processor-movingaverage` +* `snap-plugin-publisher-influxdb` -##### c) static and dynamic metrics -Snap supports both static and dynamic metrics. You can find more detail about static and dynamic metrics [here](./METRICS.md). +### Plugin Metric Namespace -### Mandatory packages -There are three mandatory packages that every plugin must use. Other than those three packages, you can use other packages as necessary. There is no danger of colliding dependencies as plugins are separated processes. The mandatory packages are: -``` -github.com/intelsdi-x/snap/control/plugin -github.com/intelsdi-x/snap/control/plugin/cpolicy -github.com/intelsdi-x/snap/core/ctypes -``` -### Writing a collector plugin -A Snap collector plugin collects telemetry data by communicating with the Snap daemon. To confine to collector plugin interfaces and metric types defined in Snap, a collector plugin must implement the following methods: -``` -GetConfigPolicy() (*cpolicy.ConfigPolicy, error) -CollectMetrics([]MetricType) ([]MetricType, error) -GetMetricTypes(ConfigType) ([]MetricType, error) -``` -The plugin uses the default values given in the ConfigPolicy so a config file doesn't need to be passed in for these rules. An example use case would be for the URL the Apache Collector collects from. Disclaimer: Two namespaces can't have rules with the same key name. E.g. you can't have the key "username" for /intel/foo/bar and a different "username" for /intel/foo/mock. They would need unique keys. +When gathering data in collector plugins, each metric requires a namespace and a description of what data is being gathered. The metric namespace should contain the org, name of plugin, and the metric's name: + +`/[organization]/[plugin name]/[plugin internal namespace(s)]/[metric name]` + +For example `ACME` org writing a `water` plugin, gather `usage` in gallons from multiple locations. -### Writing a processor plugin -A Snap processor plugin allows filtering, aggregation, transformation, etc of collected telemetry data. To complaint with processor plugin interfaces defined in Snap, a processor plugin must implement the following methods: -``` -GetConfigPolicy() (*cpolicy.ConfigPolicy, error) -Process(contentType string, content []byte, config map[string]ctypes.ConfigValue) (string, []byte, error) -``` -### Writing a publisher plugin -A Snap publisher plugin allows publishing processed telemetry data into a variety of systems, databases, and monitors through Snap metrics. To compliant with metric types and plugin interfaces defined in Snap, a publisher plugin must implement the following methods: ``` -GetConfigPolicy() (*cpolicy.ConfigPolicy, error) -Publish(contentType string, content []byte, config map[string]ctypes.ConfigValue) error +/acme/water/usage +/acme/water/NewYork/usage +/acme/water/NewYork/rainfall +/acme/water/NewYork/Bronx/usage +/acme/water/NewYork/Chelsea/usage +/acme/water/Portland/usage +/acme/water/Portland/rainfall +/acme/water/Portland/Hollywood/usage +/acme/water/Portland/Pearl/usage ``` -### Exposing a plugin -Creating the main program to serve the newly written plugin as an external process in main.go. By defining "Plugin.PluginMeta" with plugin specific settings, the newly created plugin may have its setting to override Snap global settings. Please refer to [a sample](https://github.com/intelsdi-x/snap/blob/master/plugin/collector/snap-plugin-collector-mock1/main.go) to see how main.go is written. You may browse [snap global settings](https://github.com/intelsdi-x/snap/blob/master/snapd.go#L45-L119). -Building main.go generates a binary executable. You may choose to sign the executable with our [plugin signing](https://github.com/intelsdi-x/snap/blob/master/docs/PLUGIN_SIGNING.md). +A plugin can have any number of internal plugin namespace. It's up to the plugin author on how to organize the metrics in a meaningful way. This information should be documented in the README for ease of usage. -### Localization -All comments and READMEs within the plugin code should be in English. For different languages, include appropriate translation files within the plugin package for internationalization. +NOTE: The intelsdi-x project reserves the org namespace `/intel`. -### README -All plugins should have a README with some standard fields: -``` - 1. Snap version requires at least - 2. Snap version tested up to - 3. Supported platforms - 4. Contributor - 5. License -``` -### Encryption -Snap provides the encryption capability for both HTTP and TCP clients. The communication between the Snap daemon and the plugins is encrypted by default. Should you want to disable the encrypted communication, when authoring a plugin, use the `Unsecure` option for your plugin's meta: -``` -//Meta returns the metadata for MyPlugin -func Meta() *plugin.PluginMeta { - return plugin.NewPluginMeta(name, ver, type, ct, ct2, plugin.Unsecure(true)) +### Plugin Interface + +Depending on the type of plugin, they must implement several methods to satisfy the appropriate interfaces. Please see the [plugin library](#plugin-library) for language specific examples and documentation. + +### Plugin Version + +Currently plugin versions are integer numbers and registered when a plugin is loaded. Whenever the source code is modified, please update the plugin version. + +The following plugin version is taken from [Go lib example](https://github.com/intelsdi-x/snap-plugin-lib-go/blob/master/examples/collector/main.go). + +```go +const ( + pluginName = "rand-collector" + pluginVersion = 1 +) + +func main() { + plugin.StartCollector(rand.RandCollector{}, pluginName, pluginVersion) } ``` -## Logging and debugging -Snap uses [logrus](http://github.com/Sirupsen/logrus) to log. Your plugins can use it, or any standard Go log package. Each plugin has its log file. If no logging directory is specified, logs are in the /tmp directory of the running machine. INFO is the logging level for the release version of plugins. Loggers are excellent resources for debugging. You can also use Go GDB or [delve](https://github.com/derekparker/delve) to debug. +Whenever the version changes, we also recommend: + +* git tag the repository with the new plugin version +* publish binaries in github release page -## Building and running the tests -While developing a plugin, unit and integration tests need to be performed. Snap uses [goconvey](http://github.com/smartystreets/goconvey/convey) for unit tests. You are welcome to use it or any other unit test framework. For the integration tests, you have to set up $SNAP_PATH and some necessary direct, or indirect dependencies. Using Docker container for integration tests is an effective testing strategy. Integration tests may define an input workflow. Refer to a sample [integration test input](https://github.com/intelsdi-x/snap/blob/master/examples/configs/snap-config-sample.json). +For intelsdi-x repos, the binaries publishing is automated. If we update the rand plugin to version 2, simply tag the commit with the new version and push the new tags to github: -For example, to run a plugin integration test ``` -go test -v tag=integration ./… +$ git tag -a 2 -m 'snap plugin collector rand v2' +$ git push origin --tags ``` -For more build and test tips, please refer to our [contributing doc](https://github.com/intelsdi-x/snap/blob/master/CONTRIBUTING.md). +NOTE: We are planning to adapt [Semantic Versioning](http://semver.org/). This requires changes to the internal framework, and we will provide a transition path when this is ready. + +### Plugin Release + +We recommend releasing new binaries to Github Release page whenever the plugin version is updated. This process can be automated via [Travis CI](https://docs.travis-ci.com/user/deployment/releases/). Please check out the file plugin's [.travis.yml](https://github.com/intelsdi-x/snap-plugin-publisher-file/blob/master/.travis.yml) file for a working example. + +### Plugin Metadata + +In the plugin repo root directory, the `metadata.yml` file provides Snap project additional information about your plugin when we generate the [plugin catalog](./PLUGIN_CATALOG.md) page. + +* name: plugin full name +* type: plugin type +* maintainer: your github org or username +* license: the plugin software licence +* description: paragraph describing the plugin's purpose +* badge: a list of [badges](https://shields.io/) to display +* ci: a list of ci services running for this repo + +All metadata fields are optional, but recommended to help users discover your plugin. Please check out the file plugin's [metadata.yml](https://github.com/intelsdi-x/snap-plugin-publisher-file/blob/master/metadata.yml) file for a working example. + +To list your plugin in the catalog, please submit a PR and update [plugins.yml](./plugins.yml) file to include the plugin's github `organization/repo_name`. -## Distributing plugins -If you think others would find your plugin useful, we encourage you to submit it to our [Plugin Catalog](https://github.com/intelsdi-x/snap/blob/master/docs/PLUGIN_CATALOG.md) for possible inclusion. +### Documentation -## License -The Snap framework is released under the Apache 2.0 license. +All plugins should include a README with the following information: -## For more help -Please browse more at our [repo](https://github.com/intelsdi-x/snap) or contact the [maintainers](https://github.com/intelsdi-x/snap#maintainers). +1. Supported Platforms +2. Snap Version dependencies +3. Installation +4. Usage +5. Contributors +6. License diff --git a/docs/PLUGIN_BEST_PRACTICES.md b/docs/PLUGIN_BEST_PRACTICES.md deleted file mode 100644 index cae50fad2..000000000 --- a/docs/PLUGIN_BEST_PRACTICES.md +++ /dev/null @@ -1,23 +0,0 @@ -## Best practices in plugin development: -### Leverage plugin configurability options -1. **Compile time configuration** - use `Plugin.PluginMeta` to define plugin's: name, version, type, accepted and returned content types, concurrency level, exclusiveness, secure communication settings and cache TTL. This type of configuration is usually specified in `main()` in which `plugin.Start()` method is called. -2. **Run time configuration** - - **Global** - This config is useful if configuration data is needed to obtain the list of metrics (for example: user names, paths to tools, etc.). Values from Global config (as defined in config json) are available in `GetMetricTypes()` method. - - **Task level** - This config is useful when you need to pass configuration per metric or plugin in order to collect the metrics. Use `GetConfigPolicy()` to set configurable items for plugin. Values from Task config are available in `CollectMetrics()` method. - -### Use `snap-plugin-utilities` library -The library and guide are available [here](https://github.com/intelsdi-x/snap-plugin-utilities). The library consists of the following helper packages: -* **`config`** - The config package provides helpful methods to retrieve global config items. -* **`logger`** - The logger package wraps the logrus package. (https://github.com/Sirupsen/logrus). It sets logging from a plugin to separate files and adds a caller function name to each message. It's best to use log level defined during `snapd` start. -* **`ns`** - The ns package provides functions to extract namespace from maps, JSON and struct compositions. It is useful for situations when full knowledge of available metrics is not known at the time when `GetMetricTypes()` is called. -* **`pipeline`** - Creates an array of Pipes connected by channels. Each Pipe can perform a single process on data transmitted by channels. -* **`source`** - The source package provides handy ways of dealing with external command output. It can be used for continuous command execution (PCM like), or for single command calls. -* **`stack`** - The stack package provides a simple implementation of a stack. - -### Namespace definition -* The `GetMetricTypes()` method returns namespaces for your metrics. For new metrics, it is a good idea to start with your organization (for example: "intel"), then enumerate information starting from most general to most detailed. Examples: `/intel/server/mem/free` or `/intel/linux/iostat/avg-cpu/%idle`. -* Do not use any special character in namespaces other than: "%", "-" and "_". -* When enumerating metric sources, it is always better to have them separated. For example: use `cpu/0` rather than `cpu0`. - -### Testability -It is highly recommended to deliver unit and integration tests with your code - use best practices for your language of choice to create testable code (for example in golang consider using dependency injection and interfaces).