OpenCost plugins make extending OpenCost’s coverage into new external cost sources (e.g. monitoring, data platforms, cloud services, and other SaaS solutions) available to the open source community. They allow for the ingestion of arbitrary cost data into OpenCost through conformance to the FOCUS spec.
Any plugin released within this repository can be deployed alongside OpenCost. When deployed, OpenCost will query all installed plugins with a request for the following:
- Start - timestamp
- End - timestamp
- Resolution - either hour or day
The plugins are expected to return a response that conforms to the aforementioned FOCUS spec for the given window and resolution. OpenCost will then store the response, and continue to request data for further time ranges.
The FOCUS spec is broken up into two parts:
CustomCost
CustomCostExtendedAttributes
Plugin development only necessitates the implementation of theCustomCost
response. Extended attributes are an optional response object. However, we high encourage the use of extended attributes, as doing so will assist us in developing the UX of said attributes within OpenCost.
See this ExcaliDraw diagram for a more details overview of the plugin system
At the most basic level, all a plugin needs to do is gather cost data given a time window and resolution. The logistics of this are straightforward, but the complexity of the implementation will depend on the data source in question.
- Pull the repo locally.
- Create a directory within the repo for the new plugin. Check out the Datadog plugin for a reference to follow along with.
- Create the plugin subdirectories:
<repo>/<plugin>/cmd/main/
- This will contain the actual logic of the plugin.
<repo>/<plugin>/<plugin>plugin/
- This will contain the plugin config struct and any other structs for handling requests/responses in the plugin.
<repo>/<plugin>/tests/
- Highly recommended, this will contain tests to validate the functionality of the plugin.
- Initialize the subproject:
- Within
<repo>/<plugin>/
, rungo mod init github.com/opencost/opencost-plugins/<plugin>
andgo get github.com/hashicorp/go-plugin
.
- Within
All plugins require a configuration. For example, the Datadog plugin configuration takes in some information required to authenticate with the Datadog API. This configuration will be defined by a struct inside <repo>/<plugin>/<plugin>plugin/
.
Once the configuration is designed, it's time to write the plugin. Within <repo>/<plugin>/cmd/main/>
, create main.go
:
- Create a
<plugin>Source
struct (Datadog reference). - Implement the
CustomCostSource
interface for your plugin source (Datadog reference). At the time of the writing of this guide, the only required function isGetCustomCosts
, which takes in aCustomCostRequest
object and returns a list ofCustomCostResponse
objects. Let's step through the Datadog reference implementation:- First, we split the requested window into sub-windows given the requested resolution. OpenCost has a convenience function to perform this split for us (
GetWindows
). - Next, we grab some pricing data from the Datadog API to prepare for the next step.
- Penultimately, we iterate through each sub-window, grabbing the cost data from the Datadog API for each one.
- Optionally (but recommended), implement your response data such that the custom cost extended attributes are utilized as much as possible.
- Finally, we return the retrieved cost data.
- First, we split the requested window into sub-windows given the requested resolution. OpenCost has a convenience function to perform this split for us (
- Implement the
main
function:- Find the config file (Datadog reference).
- Load the config file (Datadog reference).
- Instantiate the plugin source (Datadog reference).
- Serve the plugin for consumption by OpenCost (Datadog reference).
Write some unit tests to validate the functionality of your new plugin. See the Datadog unit tests for reference.
Now that your plugin is implemented and tested, all that's left is to get it submitted for review. Create a PR based off your branch and submit it, and an OpenCost developer will review it for you.
- OpenCost stores the plugin responses in an in-memory repository, which necessitates that OpenCost queries the plugins again for cost data upon pod restart.
- Many cost sources have API rate limits, such as Datadog. As such, a rate limiter may be necessary.
- If you want a plugin embedded in your OpenCost image, you will have to build the image yourself.
Thanks to all the individuals who have given their time and effort towards creating and maintaining these plugins:
Sajit Mathew Kunnumkal 💻 MongoDB Atlas Plugin |