Commit 6b29c117 by Marcus Olsson Committed by GitHub

Add guide for streaming data sources (#24786)

parent c5950d63
...@@ -42,8 +42,10 @@ Ready to learn more? Check out our other tutorials: ...@@ -42,8 +42,10 @@ Ready to learn more? Check out our other tutorials:
Improve an existing plugin with one of our guides: Improve an existing plugin with one of our guides:
- [Add support for annotations]({{< relref "add-support-for-annotations.md" >}}) - [Add support for annotations]({{< relref "add-support-for-annotations.md" >}})
- [Add support for Explore queries]({{< relref "add-support-for-explore-queries.md" >}})
- [Add support for variables]({{< relref "add-support-for-variables.md" >}}) - [Add support for variables]({{< relref "add-support-for-variables.md" >}})
- [Build a logs data source]({{< relref "build-a-logs-data-source.md" >}}) - [Build a logs data source plugin]({{< relref "build-a-logs-data-source-plugin.md" >}})
- [Build a streaming data source plugin]({{< relref "build-a-streaming-data-source-plugin.md" >}})
### Concepts ### Concepts
......
+++ +++
title = "Build a logs data source" title = "Build a logs data source plugin"
type = "docs" type = "docs"
+++ +++
# Build a logs data source # Build a logs data source plugin
This guide explains how to build a logs data source plugin. This guide explains how to build a logs data source plugin.
Data sources in Grafana supports both metrics and log data. The steps to build a logs data source are largely the same as for a metrics data source. This guide assumes that you're already familiar with how to [Build a data source plugin]({{< relref "../../../../../tutorials/build-a-data-source-plugin.md" >}}) for metrics. Data sources in Grafana supports both metrics and log data. The steps to build a logs data source plugin are largely the same as for a metrics data source. This guide assumes that you're already familiar with how to [Build a data source plugin]({{< relref "../../../../../tutorials/build-a-data-source-plugin.md" >}}) for metrics.
## Add logs support to your data source ## Add logs support to your data source
......
+++
title = "Build a streaming data source plugin"
type = "docs"
+++
# Build a streaming data source plugin
This guide explains how to build a streaming data source plugin.
This guide assumes that you're already familiar with how to [Build a data source plugin]({{< relref "../../../../../tutorials/build-a-data-source-plugin.md" >}}).
When monitoring critical applications, you want your dashboard to refresh as soon as your data does. In Grafana, you can set your dashboards to automatically refresh at a certain interval, no matter what data source you use. Unfortunately, this means that your queries are requesting all the data to be sent again, regardless of whether the data has actually changed.
By enabling _streaming_ for your data source plugin, you can update your dashboard as soon as new data becomes available.
For example, a streaming data source plugin can connect to a websocket, or subscribe to a message bus, and update the visualization whenever a new message is available.
Let's see how you can add streaming to an existing data source!
Grafana uses [RxJS](https://rxjs.dev/) to continuously send data from a data source to a panel visualization. There's a lot more to RxJS than what's covered in this guide. If you want to learn more, check out the [RxJS documentation](https://rxjs.dev/guide/overview).
1. Enable streaming for your data source in the `plugin.json` file.
```json
{
"streaming": true
}
```
1. Change the signature of the `query` method to return an `Observable` from the `rxjs` package. Make sure you remove the `async` keyword.
```ts
import { Observable } from 'rxjs';
```
```ts
query(options: DataQueryRequest<MyQuery>): Observable<DataQueryResponse> {
// ...
}
```
1. Create an `Observable` for each query, and then combine them all using the `merge` function from the `rxjs` package.
```ts
import { Observable, merge } from 'rxjs';
```
```ts
const observables = options.targets.map(target => {
return new Observable<DataQueryResponse>(subscriber => {
// ...
});
});
return merge(...observables);
```
1. In the `subscribe` function, create a `CircularDataFrame`.
```ts
import { CircularDataFrame } from '@grafana/data';
```
```ts
const frame = new CircularDataFrame({
append: 'tail',
capacity: 1000,
});
frame.refId = query.refId;
frame.addField({ name: 'time', type: FieldType.time });
frame.addField({ name: 'value', type: FieldType.number });
```
Circular data frames have a limited capacity. When a circular data frame reaches its capacity, the oldest data point is removed.
1. Use `subscriber.next()` to send the updated data frame whenever you receive new updates.
```ts
const intervalId = setInterval(() => {
frame.add({ time: Date.now(), value: Math.random() });
subscriber.next({
data: [frame],
key: query.refId,
});
}, 500);
return () => {
clearInterval(intervalId);
};
```
> **Note:** In practice, you'd call `subscriber.next` as soon as you receive new data from a websocket or a message bus. The example above simulates data being received every 500 milliseconds.
Here's the final `query` method.
```ts
query(options: DataQueryRequest<MyQuery>): Observable<DataQueryResponse> {
const streams = options.targets.map(target => {
const query = defaults(target, defaultQuery);
return new Observable<DataQueryResponse>(subscriber => {
const frame = new CircularDataFrame({
append: 'tail',
capacity: 1000,
});
frame.refId = query.refId;
frame.addField({ name: 'time', type: FieldType.time });
frame.addField({ name: 'value', type: FieldType.number });
const intervalId = setInterval(() => {
frame.add({ time: Date.now(), value: Math.random() });
subscriber.next({
data: [frame],
key: query.refId,
});
}, 100);
return () => {
clearInterval(intervalId);
};
});
});
return merge(...streams);
}
```
One limitation with this example is that the panel visualization is cleared every time you update the dashboard. If you have access to historical data, you can add, or _backfill_, it to the data frame before the first call to `subscriber.next()`.
...@@ -398,8 +398,10 @@ ...@@ -398,8 +398,10 @@
link: /developers/plugins/add-support-for-annotations/ link: /developers/plugins/add-support-for-annotations/
- name: Add support for Explore queries - name: Add support for Explore queries
link: /developers/plugins/add-support-for-explore-queries/ link: /developers/plugins/add-support-for-explore-queries/
- name: Build a logs data source - name: Build a logs data source plugin
link: /developers/plugins/build-a-logs-data-source/ link: /developers/plugins/build-a-logs-data-source-plugin/
- name: Build a streaming data source plugin
link: /developers/plugins/build-a-streaming-data-source-plugin/
- name: Authentication - name: Authentication
link: /developers/plugins/authentication/ link: /developers/plugins/authentication/
- name: Backend plugins - name: Backend plugins
......
Markdown is supported
0% or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment