From e6453198a91b5998f8e6a6194f853fbdfa3b7a4f Mon Sep 17 00:00:00 2001 From: Marcus Olsson Date: Wed, 16 Sep 2020 14:39:46 +0200 Subject: [PATCH] Docs: Add data frames plugin guide (#27430) --- docs/sources/developers/plugins/_index.md | 1 + .../plugins/working-with-data-frames.md | 123 ++++++++++++++++++ docs/sources/menu.yaml | 2 + 3 files changed, 126 insertions(+) create mode 100644 docs/sources/developers/plugins/working-with-data-frames.md diff --git a/docs/sources/developers/plugins/_index.md b/docs/sources/developers/plugins/_index.md index 15d7d7ebd0a72..61911173cfe41 100644 --- a/docs/sources/developers/plugins/_index.md +++ b/docs/sources/developers/plugins/_index.md @@ -49,6 +49,7 @@ Improve an existing plugin with one of our guides: - [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" >}}) - [Error handling]({{< relref "error-handling.md" >}}) +- [Working with data frames]({{< relref "working-with-data-frames.md" >}}) ### Concepts diff --git a/docs/sources/developers/plugins/working-with-data-frames.md b/docs/sources/developers/plugins/working-with-data-frames.md new file mode 100644 index 0000000000000..847c792845e21 --- /dev/null +++ b/docs/sources/developers/plugins/working-with-data-frames.md @@ -0,0 +1,123 @@ ++++ +title = "Working with data frames" +type = "docs" ++++ + +# Working with data frames + +The data frame is a columnar data structure which allows efficient querying of large amounts of data. Since data frames are a central concept when developing plugins for Grafana, in this guide we'll look at some ways you can use them. + +The [DataFrame]({{< relref "../../packages_api/data/dataframe.md" >}}>) interface contains a `name` and an array of `fields` where each field contains the name, type, and the values for the field. + +> **Note:** If you're looking to migrate an existing plugin to use the data frame format, refer to [Migrate to data frames]({{< relref "migration-guide.md#migrate-to-data-frames" >}}). + +## Create a data frame + +If you build a data source plugin, then you'll most likely want to convert a response from an external API to a data frame. Let's look at how to create a data frame. + +Let's start with creating a simple data frame that represents a time series. The easiest way to create a data frame is to use the [toDataFrame]({{< relref "../../packages_api/data/todataframe.md" >}}) function. + +```ts +// Need to be of the same length. +const timeValues = [1599471973065, 1599471975729]; +const numberValues = [12.3, 28.6]; + +// Create data frame from values. +const frame = toDataFrame({ + name: "http_requests_total", + fields: [ + { name: "Time", type: FieldType.time, values: timeValues }, + { name: "Value", type: FieldType.number, values: numberValues } + ] +}); +``` + +> **Note:** Data frames representing time series contain at least a `time` field, and a `number` field. By convention, built-in plugins use `Time` and `Value` as field names for data frames containing time series data. + +As you can see from the example, creating data frames like this requires that your data is already stored as columnar data. If you already have the records in the form of an array of objects, then you can pass it to `toDataFrame` which tries to guess the schema based on the types and names of the objects in the array. If you're creating complex data frames this way, then be sure to verify that you get the schema you expect. + +```ts +const series = [ + { Time: 1599471973065, Value: 12.3 }, + { Time: 1599471975729, Value: 28.6 } +]; + +const frame = toDataFrame(series); +frame.name = 'http_requests_total' +``` + +## Read values from a data frame + +When you're building a panel plugin, the data frames returned by the data source are available from the `data` prop in your panel component. + +```ts +const SimplePanel: React.FC = ({ data }) => { + const frame = data.series[0]; + + // ... +} +``` + +Before you start reading the data, think about what data you expect. For example, to visualize a time series we'd need at least one time field, and one number field. + +```ts +const timeField = frame.fields.find(field => field.type === FieldType.time); +const valueField = frame.fields.find(field => field.type === FieldType.number); +``` + +Other types of visualizations might need multiple dimensions. For example, a bubble chart that uses three numeric fields: the X-axis, Y-axis, and one for the radius of each bubble. In this case, instead of hard coding the field names, we recommend that you let the user choose the field to use for each dimension. + +```ts +const x = frame.fields.find(field => field.name === xField); +const y = frame.fields.find(field => field.name === yField); +const size = frame.fields.find(field => field.name === sizeField); + +for (let i = 0; i < frame.length; i++) { + const row = [x?.values.get(i), y?.values.get(i), size?.values.get(i)]; + + // ... +} +``` + +Alternatively, you can use the [DataFrameView]({{< relref "../../packages_api/data/dataframeview.md" >}}), which gives you an array of objects that contain a property for each field in the frame. + +```ts +const view = new DataFrameView(frame); + +view.forEach(row => { + console.log(row[options.xField], row[options.yField], row[options.sizeField]); +}) +``` + +## Display values from a data frame + +[Field options]({{< relref "../../panels/field-options.md" >}}) let the user control how Grafana displays the data in a data frame. + +To apply the field options to a value, use the `display` method on the corresponding field. The result contains information such as the color and suffix to use when display the value. + +```ts +const valueField = frame.fields.find(field => field.type === FieldType.number); + +return ( +
+ {valueField + ? valueField.values.toArray().map(value => { + const displayValue = valueField.display!(value); + return ( +

+ {displayValue.text} {displayValue.suffix ? displayValue.suffix : ''} +

+ ); + }) + : null} +
+); +``` + +To apply field options to the name of a field, use [getFieldDisplayName]({{< relref "../../packages_api/data/getfielddisplayname.md" >}}). + +```ts +const valueField = frame.fields.find(field => field.type === FieldType.number); +const valueFieldName = getFieldDisplayName(valueField, frame); +``` + diff --git a/docs/sources/menu.yaml b/docs/sources/menu.yaml index 424229b943998..be484e12e81ee 100644 --- a/docs/sources/menu.yaml +++ b/docs/sources/menu.yaml @@ -466,6 +466,8 @@ link: /developers/plugins/metadata/ - name: Data frames link: /developers/plugins/data-frames/ + - name: Working with data frames + link: /developers/plugins/working-with-data-frames/ - name: Add support for variables link: /developers/plugins/add-support-for-variables/ - name: Add support for annotations