mirror of
https://github.com/grafana/grafana.git
synced 2025-02-25 18:55:37 -06:00
Transforms: Add join by fields (#61322)
This commit is contained in:
parent
f7d92ab841
commit
bfbc8c3c4f
@ -24,7 +24,6 @@
|
|||||||
"editable": true,
|
"editable": true,
|
||||||
"fiscalYearStartMonth": 0,
|
"fiscalYearStartMonth": 0,
|
||||||
"graphTooltip": 0,
|
"graphTooltip": 0,
|
||||||
"id": 1351,
|
|
||||||
"links": [],
|
"links": [],
|
||||||
"liveNow": false,
|
"liveNow": false,
|
||||||
"panels": [
|
"panels": [
|
||||||
@ -38,7 +37,7 @@
|
|||||||
},
|
},
|
||||||
"id": 9,
|
"id": 9,
|
||||||
"panels": [],
|
"panels": [],
|
||||||
"title": "Join by time",
|
"title": "Input",
|
||||||
"type": "row"
|
"type": "row"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -49,37 +48,14 @@
|
|||||||
"fieldConfig": {
|
"fieldConfig": {
|
||||||
"defaults": {
|
"defaults": {
|
||||||
"color": {
|
"color": {
|
||||||
"mode": "palette-classic"
|
"mode": "thresholds"
|
||||||
},
|
},
|
||||||
"custom": {
|
"custom": {
|
||||||
"axisCenteredZero": false,
|
"align": "auto",
|
||||||
"axisColorMode": "text",
|
"cellOptions": {
|
||||||
"axisLabel": "",
|
"type": "auto"
|
||||||
"axisPlacement": "auto",
|
|
||||||
"barAlignment": 0,
|
|
||||||
"drawStyle": "line",
|
|
||||||
"fillOpacity": 0,
|
|
||||||
"gradientMode": "none",
|
|
||||||
"hideFrom": {
|
|
||||||
"legend": false,
|
|
||||||
"tooltip": false,
|
|
||||||
"viz": false
|
|
||||||
},
|
},
|
||||||
"lineInterpolation": "linear",
|
"inspect": false
|
||||||
"lineWidth": 1,
|
|
||||||
"pointSize": 5,
|
|
||||||
"scaleDistribution": {
|
|
||||||
"type": "linear"
|
|
||||||
},
|
|
||||||
"showPoints": "auto",
|
|
||||||
"spanNulls": false,
|
|
||||||
"stacking": {
|
|
||||||
"group": "A",
|
|
||||||
"mode": "none"
|
|
||||||
},
|
|
||||||
"thresholdsStyle": {
|
|
||||||
"mode": "off"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"mappings": [],
|
"mappings": [],
|
||||||
"thresholds": {
|
"thresholds": {
|
||||||
@ -99,41 +75,41 @@
|
|||||||
},
|
},
|
||||||
"gridPos": {
|
"gridPos": {
|
||||||
"h": 8,
|
"h": 8,
|
||||||
"w": 12,
|
"w": 8,
|
||||||
"x": 0,
|
"x": 0,
|
||||||
"y": 1
|
"y": 1
|
||||||
},
|
},
|
||||||
"id": 11,
|
"id": 11,
|
||||||
"options": {
|
"options": {
|
||||||
"legend": {
|
"footer": {
|
||||||
"calcs": [],
|
"countRows": false,
|
||||||
"displayMode": "list",
|
"fields": "",
|
||||||
"placement": "bottom",
|
"reducer": [
|
||||||
"showLegend": true
|
"sum"
|
||||||
|
],
|
||||||
|
"show": false
|
||||||
},
|
},
|
||||||
"tooltip": {
|
"showHeader": true
|
||||||
"mode": "single",
|
|
||||||
"sort": "none"
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
|
"pluginVersion": "9.4.0-pre",
|
||||||
"targets": [
|
"targets": [
|
||||||
{
|
{
|
||||||
"datasource": {
|
"datasource": {
|
||||||
"type": "testdata",
|
"type": "testdata",
|
||||||
"uid": "PD8C576611E62080A"
|
"uid": "PD8C576611E62080A"
|
||||||
},
|
},
|
||||||
"refId": "A",
|
"rawFrameContent": "[{\r\n \"name\": \"tags\",\r\n \"fields\": [\r\n { \"name\": \"tags__time\", \"values\": [100, 101, 200] },\r\n { \"name\": \"tags__name\", \"values\": [\"v1.2\", \"v1.2b\", \"v1.3\"] }\r\n ]\r\n}]",
|
||||||
"scenarioId": "random_walk",
|
"refId": "tags",
|
||||||
"seriesCount": 4
|
"scenarioId": "raw_frame"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"title": "Timeseries data",
|
"title": "tags",
|
||||||
"type": "timeseries"
|
"type": "table"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"datasource": {
|
"datasource": {
|
||||||
"type": "datasource",
|
"type": "testdata",
|
||||||
"uid": "-- Dashboard --"
|
"uid": "PD8C576611E62080A"
|
||||||
},
|
},
|
||||||
"fieldConfig": {
|
"fieldConfig": {
|
||||||
"defaults": {
|
"defaults": {
|
||||||
@ -142,7 +118,9 @@
|
|||||||
},
|
},
|
||||||
"custom": {
|
"custom": {
|
||||||
"align": "auto",
|
"align": "auto",
|
||||||
"displayMode": "auto",
|
"cellOptions": {
|
||||||
|
"type": "auto"
|
||||||
|
},
|
||||||
"inspect": false
|
"inspect": false
|
||||||
},
|
},
|
||||||
"mappings": [],
|
"mappings": [],
|
||||||
@ -163,13 +141,14 @@
|
|||||||
},
|
},
|
||||||
"gridPos": {
|
"gridPos": {
|
||||||
"h": 8,
|
"h": 8,
|
||||||
"w": 12,
|
"w": 8,
|
||||||
"x": 12,
|
"x": 8,
|
||||||
"y": 1
|
"y": 1
|
||||||
},
|
},
|
||||||
"id": 13,
|
"id": 13,
|
||||||
"options": {
|
"options": {
|
||||||
"footer": {
|
"footer": {
|
||||||
|
"countRows": false,
|
||||||
"fields": "",
|
"fields": "",
|
||||||
"reducer": [
|
"reducer": [
|
||||||
"sum"
|
"sum"
|
||||||
@ -178,24 +157,25 @@
|
|||||||
},
|
},
|
||||||
"showHeader": true
|
"showHeader": true
|
||||||
},
|
},
|
||||||
"pluginVersion": "9.2.0-pre",
|
"pluginVersion": "9.4.0-pre",
|
||||||
"targets": [
|
"targets": [
|
||||||
{
|
{
|
||||||
"datasource": {
|
"datasource": {
|
||||||
"type": "datasource",
|
"type": "testdata",
|
||||||
"uid": "-- Dashboard --"
|
"uid": "PD8C576611E62080A"
|
||||||
},
|
},
|
||||||
"panelId": 11,
|
"rawFrameContent": "[{\r\n \"name\": \"releases\",\r\n\"fields\": [\r\n { \"name\": \"releases__time\", \"values\": [150, 250] },\r\n { \"name\": \"releases__tag\", \"values\": [\"v1.2\", \"v1.3\"] }\r\n]}]",
|
||||||
"refId": "A"
|
"refId": "releases",
|
||||||
|
"scenarioId": "raw_frame"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"title": "Same data (as a table)",
|
"title": "releases",
|
||||||
"type": "table"
|
"type": "table"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"datasource": {
|
"datasource": {
|
||||||
"type": "datasource",
|
"type": "testdata",
|
||||||
"uid": "-- Dashboard --"
|
"uid": "PD8C576611E62080A"
|
||||||
},
|
},
|
||||||
"fieldConfig": {
|
"fieldConfig": {
|
||||||
"defaults": {
|
"defaults": {
|
||||||
@ -204,7 +184,9 @@
|
|||||||
},
|
},
|
||||||
"custom": {
|
"custom": {
|
||||||
"align": "auto",
|
"align": "auto",
|
||||||
"displayMode": "auto",
|
"cellOptions": {
|
||||||
|
"type": "auto"
|
||||||
|
},
|
||||||
"inspect": false
|
"inspect": false
|
||||||
},
|
},
|
||||||
"mappings": [],
|
"mappings": [],
|
||||||
@ -224,14 +206,15 @@
|
|||||||
"overrides": []
|
"overrides": []
|
||||||
},
|
},
|
||||||
"gridPos": {
|
"gridPos": {
|
||||||
"h": 5,
|
"h": 8,
|
||||||
"w": 24,
|
"w": 8,
|
||||||
"x": 0,
|
"x": 16,
|
||||||
"y": 9
|
"y": 1
|
||||||
},
|
},
|
||||||
"id": 16,
|
"id": 19,
|
||||||
"options": {
|
"options": {
|
||||||
"footer": {
|
"footer": {
|
||||||
|
"countRows": false,
|
||||||
"fields": "",
|
"fields": "",
|
||||||
"reducer": [
|
"reducer": [
|
||||||
"sum"
|
"sum"
|
||||||
@ -240,24 +223,19 @@
|
|||||||
},
|
},
|
||||||
"showHeader": true
|
"showHeader": true
|
||||||
},
|
},
|
||||||
"pluginVersion": "9.2.0-pre",
|
"pluginVersion": "9.4.0-pre",
|
||||||
"targets": [
|
"targets": [
|
||||||
{
|
{
|
||||||
"datasource": {
|
"datasource": {
|
||||||
"type": "datasource",
|
"type": "testdata",
|
||||||
"uid": "-- Dashboard --"
|
"uid": "PD8C576611E62080A"
|
||||||
},
|
},
|
||||||
"panelId": 11,
|
"rawFrameContent": "[{\r\n \"name\": \"features\",\r\n\"fields\": [\r\n { \"name\": \"features__name\", \"values\": [\"A\", \"B\", \"C\", \"D\", \"E\"] },\r\n { \"name\": \"features__tag\", \"values\": [\"v1.2\", \"v1.3\", \"v1.2b\", \"v1.3\", \"v1.2\"] }\r\n]}]",
|
||||||
"refId": "A"
|
"refId": "features",
|
||||||
}
|
"scenarioId": "raw_frame"
|
||||||
],
|
|
||||||
"title": "OUTER join on time (default)",
|
|
||||||
"transformations": [
|
|
||||||
{
|
|
||||||
"id": "joinByField",
|
|
||||||
"options": {}
|
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
"title": "features",
|
||||||
"type": "table"
|
"type": "table"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -266,11 +244,11 @@
|
|||||||
"h": 1,
|
"h": 1,
|
||||||
"w": 24,
|
"w": 24,
|
||||||
"x": 0,
|
"x": 0,
|
||||||
"y": 14
|
"y": 9
|
||||||
},
|
},
|
||||||
"id": 5,
|
"id": 21,
|
||||||
"panels": [],
|
"panels": [],
|
||||||
"title": "Join by string field",
|
"title": "Output",
|
||||||
"type": "row"
|
"type": "row"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
@ -285,7 +263,9 @@
|
|||||||
},
|
},
|
||||||
"custom": {
|
"custom": {
|
||||||
"align": "auto",
|
"align": "auto",
|
||||||
"displayMode": "auto",
|
"cellOptions": {
|
||||||
|
"type": "auto"
|
||||||
|
},
|
||||||
"inspect": false
|
"inspect": false
|
||||||
},
|
},
|
||||||
"mappings": [],
|
"mappings": [],
|
||||||
@ -308,202 +288,61 @@
|
|||||||
"h": 8,
|
"h": 8,
|
||||||
"w": 12,
|
"w": 12,
|
||||||
"x": 0,
|
"x": 0,
|
||||||
"y": 15
|
"y": 10
|
||||||
},
|
},
|
||||||
"id": 2,
|
"id": 23,
|
||||||
"options": {
|
"options": {
|
||||||
"footer": {
|
"footer": {
|
||||||
|
"countRows": false,
|
||||||
"fields": "",
|
"fields": "",
|
||||||
"reducer": [
|
"reducer": [
|
||||||
"sum"
|
"sum"
|
||||||
],
|
],
|
||||||
"show": false
|
"show": false
|
||||||
},
|
},
|
||||||
"frameIndex": 0,
|
|
||||||
"showHeader": true
|
"showHeader": true
|
||||||
},
|
},
|
||||||
"pluginVersion": "9.2.0-pre",
|
"pluginVersion": "9.4.0-pre",
|
||||||
"targets": [
|
"targets": [
|
||||||
{
|
{
|
||||||
"csvContent": "OrderID,CustomerID,Time\n100,A,10000\n101,B,20000\n102,C,30000",
|
|
||||||
"datasource": {
|
"datasource": {
|
||||||
"type": "testdata",
|
"type": "testdata",
|
||||||
"uid": "PD8C576611E62080A"
|
"uid": "PD8C576611E62080A"
|
||||||
},
|
},
|
||||||
"refId": "Orders",
|
"rawFrameContent": "[{\r\n \"name\": \"tags\",\r\n \"fields\": [\r\n { \"name\": \"tags__time\", \"values\": [100, 101, 200] },\r\n { \"name\": \"tags__name\", \"values\": [\"v1.2\", \"v1.2b\", \"v1.3\"] }\r\n ]\r\n}]",
|
||||||
"scenarioId": "csv_content"
|
"refId": "tags",
|
||||||
|
"scenarioId": "raw_frame"
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
"csvContent": "CustomerID,Name,Country\nA,Customer A,USA\nB,Customer B,Germany\nC,Customer C,Spain\nD,Customer D,Canada",
|
|
||||||
"datasource": {
|
"datasource": {
|
||||||
"type": "testdata",
|
"type": "testdata",
|
||||||
"uid": "PD8C576611E62080A"
|
"uid": "PD8C576611E62080A"
|
||||||
},
|
},
|
||||||
"hide": false,
|
"rawFrameContent": "[{\r\n \"name\": \"releases\",\r\n \"fields\": [\r\n { \"name\": \"releases__time\", \"values\": [150, 250] },\r\n { \"name\": \"releases__tag\", \"values\": [\"v1.2\", \"v1.3\"] }\r\n]}]",
|
||||||
"refId": "Customers",
|
"refId": "releases",
|
||||||
"scenarioId": "csv_content"
|
"scenarioId": "raw_frame"
|
||||||
}
|
|
||||||
],
|
|
||||||
"title": "Orders",
|
|
||||||
"transformations": [],
|
|
||||||
"type": "table"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"datasource": {
|
|
||||||
"type": "datasource",
|
|
||||||
"uid": "-- Dashboard --"
|
|
||||||
},
|
|
||||||
"fieldConfig": {
|
|
||||||
"defaults": {
|
|
||||||
"color": {
|
|
||||||
"mode": "thresholds"
|
|
||||||
},
|
|
||||||
"custom": {
|
|
||||||
"align": "auto",
|
|
||||||
"displayMode": "auto",
|
|
||||||
"inspect": false
|
|
||||||
},
|
|
||||||
"mappings": [],
|
|
||||||
"thresholds": {
|
|
||||||
"mode": "absolute",
|
|
||||||
"steps": [
|
|
||||||
{
|
|
||||||
"color": "green"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"color": "red",
|
|
||||||
"value": 80
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
},
|
},
|
||||||
"overrides": []
|
|
||||||
},
|
|
||||||
"gridPos": {
|
|
||||||
"h": 8,
|
|
||||||
"w": 12,
|
|
||||||
"x": 12,
|
|
||||||
"y": 15
|
|
||||||
},
|
|
||||||
"id": 3,
|
|
||||||
"options": {
|
|
||||||
"footer": {
|
|
||||||
"fields": "",
|
|
||||||
"reducer": [
|
|
||||||
"sum"
|
|
||||||
],
|
|
||||||
"show": false
|
|
||||||
},
|
|
||||||
"frameIndex": 1,
|
|
||||||
"showHeader": true
|
|
||||||
},
|
|
||||||
"pluginVersion": "9.2.0-pre",
|
|
||||||
"targets": [
|
|
||||||
{
|
{
|
||||||
"datasource": {
|
"datasource": {
|
||||||
"type": "datasource",
|
"type": "testdata",
|
||||||
"uid": "-- Dashboard --"
|
"uid": "PD8C576611E62080A"
|
||||||
},
|
},
|
||||||
"panelId": 2,
|
"rawFrameContent": "[{\r\n \"name\": \"features\",\r\n \"fields\": [\r\n { \"name\": \"features__name\", \"values\": [\"A\", \"B\", \"C\", \"D\", \"E\"] },\r\n { \"name\": \"features__tag\", \"values\": [\"v1.2\", \"v1.3\", \"v1.2b\", \"v1.3\", \"v1.2\"] }\r\n]}]",
|
||||||
"refId": "A"
|
"refId": "features",
|
||||||
|
"scenarioId": "raw_frame"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"title": "Customers",
|
"title": "OUTER JOIN",
|
||||||
"transformations": [],
|
|
||||||
"type": "table"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"datasource": {
|
|
||||||
"type": "datasource",
|
|
||||||
"uid": "-- Dashboard --"
|
|
||||||
},
|
|
||||||
"fieldConfig": {
|
|
||||||
"defaults": {
|
|
||||||
"color": {
|
|
||||||
"mode": "thresholds"
|
|
||||||
},
|
|
||||||
"custom": {
|
|
||||||
"align": "auto",
|
|
||||||
"displayMode": "auto",
|
|
||||||
"inspect": false
|
|
||||||
},
|
|
||||||
"mappings": [],
|
|
||||||
"thresholds": {
|
|
||||||
"mode": "absolute",
|
|
||||||
"steps": [
|
|
||||||
{
|
|
||||||
"color": "green"
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"color": "red",
|
|
||||||
"value": 80
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"overrides": [
|
|
||||||
{
|
|
||||||
"matcher": {
|
|
||||||
"id": "byName",
|
|
||||||
"options": "CustomerID"
|
|
||||||
},
|
|
||||||
"properties": [
|
|
||||||
{
|
|
||||||
"id": "custom.width",
|
|
||||||
"value": 101
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"matcher": {
|
|
||||||
"id": "byName",
|
|
||||||
"options": "OrderID"
|
|
||||||
},
|
|
||||||
"properties": [
|
|
||||||
{
|
|
||||||
"id": "custom.width",
|
|
||||||
"value": 89
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
"gridPos": {
|
|
||||||
"h": 8,
|
|
||||||
"w": 12,
|
|
||||||
"x": 0,
|
|
||||||
"y": 23
|
|
||||||
},
|
|
||||||
"id": 6,
|
|
||||||
"options": {
|
|
||||||
"footer": {
|
|
||||||
"fields": "",
|
|
||||||
"reducer": [
|
|
||||||
"sum"
|
|
||||||
],
|
|
||||||
"show": false
|
|
||||||
},
|
|
||||||
"frameIndex": 0,
|
|
||||||
"showHeader": true,
|
|
||||||
"sortBy": []
|
|
||||||
},
|
|
||||||
"pluginVersion": "9.2.0-pre",
|
|
||||||
"targets": [
|
|
||||||
{
|
|
||||||
"datasource": {
|
|
||||||
"type": "datasource",
|
|
||||||
"uid": "-- Dashboard --"
|
|
||||||
},
|
|
||||||
"panelId": 2,
|
|
||||||
"refId": "A"
|
|
||||||
}
|
|
||||||
],
|
|
||||||
"title": "OUTER join on CustomerID (keeps missing values)",
|
|
||||||
"transformations": [
|
"transformations": [
|
||||||
{
|
{
|
||||||
"id": "joinByField",
|
"id": "joinByField",
|
||||||
"options": {
|
"options": {
|
||||||
"byField": "CustomerID",
|
"fields": {
|
||||||
|
"A": "features__name",
|
||||||
|
"features": "features__tag",
|
||||||
|
"releases": "releases__tag",
|
||||||
|
"tags": "tags__name"
|
||||||
|
},
|
||||||
"mode": "outer"
|
"mode": "outer"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -512,8 +351,8 @@
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
"datasource": {
|
"datasource": {
|
||||||
"type": "datasource",
|
"type": "testdata",
|
||||||
"uid": "-- Dashboard --"
|
"uid": "PD8C576611E62080A"
|
||||||
},
|
},
|
||||||
"fieldConfig": {
|
"fieldConfig": {
|
||||||
"defaults": {
|
"defaults": {
|
||||||
@ -522,7 +361,9 @@
|
|||||||
},
|
},
|
||||||
"custom": {
|
"custom": {
|
||||||
"align": "auto",
|
"align": "auto",
|
||||||
"displayMode": "auto",
|
"cellOptions": {
|
||||||
|
"type": "auto"
|
||||||
|
},
|
||||||
"inspect": false
|
"inspect": false
|
||||||
},
|
},
|
||||||
"mappings": [],
|
"mappings": [],
|
||||||
@ -539,69 +380,67 @@
|
|||||||
]
|
]
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"overrides": [
|
"overrides": []
|
||||||
{
|
|
||||||
"matcher": {
|
|
||||||
"id": "byName",
|
|
||||||
"options": "CustomerID"
|
|
||||||
},
|
|
||||||
"properties": [
|
|
||||||
{
|
|
||||||
"id": "custom.width",
|
|
||||||
"value": 101
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
|
||||||
{
|
|
||||||
"matcher": {
|
|
||||||
"id": "byName",
|
|
||||||
"options": "OrderID"
|
|
||||||
},
|
|
||||||
"properties": [
|
|
||||||
{
|
|
||||||
"id": "custom.width",
|
|
||||||
"value": 89
|
|
||||||
}
|
|
||||||
]
|
|
||||||
}
|
|
||||||
]
|
|
||||||
},
|
},
|
||||||
"gridPos": {
|
"gridPos": {
|
||||||
"h": 8,
|
"h": 8,
|
||||||
"w": 12,
|
"w": 12,
|
||||||
"x": 12,
|
"x": 12,
|
||||||
"y": 23
|
"y": 10
|
||||||
},
|
},
|
||||||
"id": 7,
|
"id": 24,
|
||||||
"options": {
|
"options": {
|
||||||
"footer": {
|
"footer": {
|
||||||
|
"countRows": false,
|
||||||
"fields": "",
|
"fields": "",
|
||||||
"reducer": [
|
"reducer": [
|
||||||
"sum"
|
"sum"
|
||||||
],
|
],
|
||||||
"show": false
|
"show": false
|
||||||
},
|
},
|
||||||
"frameIndex": 0,
|
"showHeader": true
|
||||||
"showHeader": true,
|
|
||||||
"sortBy": []
|
|
||||||
},
|
},
|
||||||
"pluginVersion": "9.2.0-pre",
|
"pluginVersion": "9.4.0-pre",
|
||||||
"targets": [
|
"targets": [
|
||||||
{
|
{
|
||||||
"datasource": {
|
"datasource": {
|
||||||
"type": "datasource",
|
"type": "testdata",
|
||||||
"uid": "-- Dashboard --"
|
"uid": "PD8C576611E62080A"
|
||||||
},
|
},
|
||||||
"panelId": 2,
|
"rawFrameContent": "[{\r\n \"name\": \"tags\",\r\n \"fields\": [\r\n { \"name\": \"tags__time\", \"values\": [100, 101, 200] },\r\n { \"name\": \"tags__name\", \"values\": [\"v1.2\", \"v1.2b\", \"v1.3\"] }\r\n ]\r\n}]",
|
||||||
"refId": "A"
|
"refId": "tags",
|
||||||
|
"scenarioId": "raw_frame"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"datasource": {
|
||||||
|
"type": "testdata",
|
||||||
|
"uid": "PD8C576611E62080A"
|
||||||
|
},
|
||||||
|
"rawFrameContent": "[{\r\n \"name\": \"releases\",\r\n \"fields\": [\r\n { \"name\": \"releases__time\", \"values\": [150, 250] },\r\n { \"name\": \"releases__tag\", \"values\": [\"v1.2\", \"v1.3\"] }\r\n]}]",
|
||||||
|
"refId": "releases",
|
||||||
|
"scenarioId": "raw_frame"
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"datasource": {
|
||||||
|
"type": "testdata",
|
||||||
|
"uid": "PD8C576611E62080A"
|
||||||
|
},
|
||||||
|
"rawFrameContent": "[{\r\n \"name\": \"features\",\r\n \"fields\": [\r\n { \"name\": \"features__name\", \"values\": [\"A\", \"B\", \"C\", \"D\", \"E\"] },\r\n { \"name\": \"features__tag\", \"values\": [\"v1.2\", \"v1.3\", \"v1.2b\", \"v1.3\", \"v1.2\"] }\r\n]}]",
|
||||||
|
"refId": "features",
|
||||||
|
"scenarioId": "raw_frame"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"title": "INNER join on CustomerID ",
|
"title": "INNER JOIN",
|
||||||
"transformations": [
|
"transformations": [
|
||||||
{
|
{
|
||||||
"id": "joinByField",
|
"id": "joinByField",
|
||||||
"options": {
|
"options": {
|
||||||
"byField": "CustomerID",
|
"fields": {
|
||||||
|
"A": "features__name",
|
||||||
|
"features": "features__tag",
|
||||||
|
"releases": "releases__tag",
|
||||||
|
"tags": "tags__name"
|
||||||
|
},
|
||||||
"mode": "inner"
|
"mode": "inner"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@ -609,7 +448,8 @@
|
|||||||
"type": "table"
|
"type": "table"
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
"schemaVersion": 37,
|
"revision": 1,
|
||||||
|
"schemaVersion": 38,
|
||||||
"style": "dark",
|
"style": "dark",
|
||||||
"tags": [
|
"tags": [
|
||||||
"gdev",
|
"gdev",
|
||||||
@ -626,6 +466,6 @@
|
|||||||
"timezone": "",
|
"timezone": "",
|
||||||
"title": "Join by field",
|
"title": "Join by field",
|
||||||
"uid": "gw0K4rmVz",
|
"uid": "gw0K4rmVz",
|
||||||
"version": 6,
|
"version": 1,
|
||||||
"weekStart": ""
|
"weekStart": ""
|
||||||
}
|
}
|
@ -20,12 +20,13 @@ export const ensureColumnsTransformer: SynchronousDataTransformerInfo = {
|
|||||||
const timeFieldName = findConsistentTimeFieldName(frames);
|
const timeFieldName = findConsistentTimeFieldName(frames);
|
||||||
|
|
||||||
if (frames.length > 1 && timeFieldName) {
|
if (frames.length > 1 && timeFieldName) {
|
||||||
return joinByFieldTransformer.transformer(
|
const fields: { [key: string]: string } = {};
|
||||||
{
|
for (const frame of frames) {
|
||||||
byField: timeFieldName,
|
if (frame.refId) {
|
||||||
},
|
fields[frame.refId] = timeFieldName;
|
||||||
ctx
|
}
|
||||||
)(frames);
|
}
|
||||||
|
return joinByFieldTransformer.transformer({ fields }, ctx)(frames);
|
||||||
}
|
}
|
||||||
return frames;
|
return frames;
|
||||||
},
|
},
|
||||||
|
@ -15,6 +15,7 @@ describe('JOIN Transformer', () => {
|
|||||||
describe('outer join', () => {
|
describe('outer join', () => {
|
||||||
const everySecondSeries = toDataFrame({
|
const everySecondSeries = toDataFrame({
|
||||||
name: 'even',
|
name: 'even',
|
||||||
|
refId: 'even',
|
||||||
fields: [
|
fields: [
|
||||||
{ name: 'time', type: FieldType.time, values: [3000, 4000, 5000, 6000] },
|
{ name: 'time', type: FieldType.time, values: [3000, 4000, 5000, 6000] },
|
||||||
{ name: 'temperature', type: FieldType.number, values: [10.3, 10.4, 10.5, 10.6] },
|
{ name: 'temperature', type: FieldType.number, values: [10.3, 10.4, 10.5, 10.6] },
|
||||||
@ -24,6 +25,7 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
const everyOtherSecondSeries = toDataFrame({
|
const everyOtherSecondSeries = toDataFrame({
|
||||||
name: 'odd',
|
name: 'odd',
|
||||||
|
refId: 'odd',
|
||||||
fields: [
|
fields: [
|
||||||
{ name: 'time', type: FieldType.time, values: [1000, 3000, 5000, 7000] },
|
{ name: 'time', type: FieldType.time, values: [1000, 3000, 5000, 7000] },
|
||||||
{ name: 'temperature', type: FieldType.number, values: [11.1, 11.3, 11.5, 11.7] },
|
{ name: 'temperature', type: FieldType.number, values: [11.1, 11.3, 11.5, 11.7] },
|
||||||
@ -33,9 +35,12 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('joins by time field', async () => {
|
it('joins by time field', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'time',
|
fields: {
|
||||||
|
even: 'time',
|
||||||
|
odd: 'time',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -135,9 +140,12 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('joins by temperature field', async () => {
|
it('joins by temperature field', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'temperature',
|
fields: {
|
||||||
|
even: 'temperature',
|
||||||
|
odd: 'temperature',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -145,6 +153,7 @@ describe('JOIN Transformer', () => {
|
|||||||
(received) => {
|
(received) => {
|
||||||
const data = received[0];
|
const data = received[0];
|
||||||
const filtered = data[0];
|
const filtered = data[0];
|
||||||
|
|
||||||
expect(filtered.fields).toMatchInlineSnapshot(`
|
expect(filtered.fields).toMatchInlineSnapshot(`
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
@ -251,9 +260,12 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('joins by time field in reverse order', async () => {
|
it('joins by time field in reverse order', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'time',
|
fields: {
|
||||||
|
even: 'time',
|
||||||
|
odd: 'time',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -265,6 +277,7 @@ describe('JOIN Transformer', () => {
|
|||||||
(received) => {
|
(received) => {
|
||||||
const data = received[0];
|
const data = received[0];
|
||||||
const filtered = data[0];
|
const filtered = data[0];
|
||||||
|
|
||||||
expect(filtered.fields).toMatchInlineSnapshot(`
|
expect(filtered.fields).toMatchInlineSnapshot(`
|
||||||
[
|
[
|
||||||
{
|
{
|
||||||
@ -376,9 +389,12 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('when dataframe and field share the same name then use the field name', async () => {
|
it('when dataframe and field share the same name then use the field name', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'time',
|
fields: {
|
||||||
|
even: 'time',
|
||||||
|
odd: 'time',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -439,9 +455,12 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('joins if fields are missing', async () => {
|
it('joins if fields are missing', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'time',
|
fields: {
|
||||||
|
even: 'time',
|
||||||
|
odd: 'time',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -517,9 +536,12 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('handles duplicate field name', async () => {
|
it('handles duplicate field name', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'time',
|
fields: {
|
||||||
|
even: 'time',
|
||||||
|
odd: 'time',
|
||||||
|
},
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
@ -580,6 +602,7 @@ describe('JOIN Transformer', () => {
|
|||||||
describe('inner join', () => {
|
describe('inner join', () => {
|
||||||
const seriesA = toDataFrame({
|
const seriesA = toDataFrame({
|
||||||
name: 'A',
|
name: 'A',
|
||||||
|
refId: 'A',
|
||||||
fields: [
|
fields: [
|
||||||
{ name: 'time', type: FieldType.time, values: [3000, 4000, 5000, 6000] },
|
{ name: 'time', type: FieldType.time, values: [3000, 4000, 5000, 6000] },
|
||||||
{ name: 'temperature', type: FieldType.number, values: [10.3, 10.4, 10.5, 10.6] },
|
{ name: 'temperature', type: FieldType.number, values: [10.3, 10.4, 10.5, 10.6] },
|
||||||
@ -589,6 +612,7 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
const seriesB = toDataFrame({
|
const seriesB = toDataFrame({
|
||||||
name: 'B',
|
name: 'B',
|
||||||
|
refId: 'B',
|
||||||
fields: [
|
fields: [
|
||||||
{ name: 'time', type: FieldType.time, values: [1000, 3000, 5000, 7000] },
|
{ name: 'time', type: FieldType.time, values: [1000, 3000, 5000, 7000] },
|
||||||
{ name: 'temperature', type: FieldType.number, values: [11.1, 10.3, 10.5, 11.7] },
|
{ name: 'temperature', type: FieldType.number, values: [11.1, 10.3, 10.5, 11.7] },
|
||||||
@ -598,9 +622,12 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('inner joins by time field', async () => {
|
it('inner joins by time field', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'time',
|
fields: {
|
||||||
|
A: 'time',
|
||||||
|
B: 'time',
|
||||||
|
},
|
||||||
mode: JoinMode.inner,
|
mode: JoinMode.inner,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
@ -679,9 +706,12 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('inner joins by temperature field', async () => {
|
it('inner joins by temperature field', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'temperature',
|
fields: {
|
||||||
|
A: 'temperature',
|
||||||
|
B: 'temperature',
|
||||||
|
},
|
||||||
mode: JoinMode.inner,
|
mode: JoinMode.inner,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
@ -764,9 +794,12 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('inner joins by time field in reverse order', async () => {
|
it('inner joins by time field in reverse order', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'time',
|
fields: {
|
||||||
|
A: 'time',
|
||||||
|
B: 'time',
|
||||||
|
},
|
||||||
mode: JoinMode.inner,
|
mode: JoinMode.inner,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
@ -852,6 +885,7 @@ describe('JOIN Transformer', () => {
|
|||||||
describe('Field names', () => {
|
describe('Field names', () => {
|
||||||
const seriesWithSameFieldAndDataFrameName = toDataFrame({
|
const seriesWithSameFieldAndDataFrameName = toDataFrame({
|
||||||
name: 'temperature',
|
name: 'temperature',
|
||||||
|
refId: 'temperature',
|
||||||
fields: [
|
fields: [
|
||||||
{ name: 'time', type: FieldType.time, values: [1000, 2000, 3000, 4000] },
|
{ name: 'time', type: FieldType.time, values: [1000, 2000, 3000, 4000] },
|
||||||
{ name: 'temperature', type: FieldType.number, values: [1, 3, 5, 7] },
|
{ name: 'temperature', type: FieldType.number, values: [1, 3, 5, 7] },
|
||||||
@ -860,6 +894,7 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
const seriesB = toDataFrame({
|
const seriesB = toDataFrame({
|
||||||
name: 'B',
|
name: 'B',
|
||||||
|
refId: 'B',
|
||||||
fields: [
|
fields: [
|
||||||
{ name: 'time', type: FieldType.time, values: [1000, 2000, 3000, 4000] },
|
{ name: 'time', type: FieldType.time, values: [1000, 2000, 3000, 4000] },
|
||||||
{ name: 'temperature', type: FieldType.number, values: [2, 4, 6, 8] },
|
{ name: 'temperature', type: FieldType.number, values: [2, 4, 6, 8] },
|
||||||
@ -868,9 +903,12 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('when dataframe and field share the same name then use the field name', async () => {
|
it('when dataframe and field share the same name then use the field name', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'time',
|
fields: {
|
||||||
|
temperature: 'time',
|
||||||
|
B: 'time',
|
||||||
|
},
|
||||||
mode: JoinMode.inner,
|
mode: JoinMode.inner,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
@ -932,15 +970,20 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('joins if fields are missing', async () => {
|
it('joins if fields are missing', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'time',
|
fields: {
|
||||||
|
A: 'time',
|
||||||
|
B: 'time',
|
||||||
|
C: 'time',
|
||||||
|
},
|
||||||
mode: JoinMode.inner,
|
mode: JoinMode.inner,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const frame1 = toDataFrame({
|
const frame1 = toDataFrame({
|
||||||
name: 'A',
|
name: 'A',
|
||||||
|
refId: 'A',
|
||||||
fields: [
|
fields: [
|
||||||
{ name: 'time', type: FieldType.time, values: [1, 2, 3] },
|
{ name: 'time', type: FieldType.time, values: [1, 2, 3] },
|
||||||
{ name: 'temperature', type: FieldType.number, values: [10, 11, 12] },
|
{ name: 'temperature', type: FieldType.number, values: [10, 11, 12] },
|
||||||
@ -949,11 +992,13 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
const frame2 = toDataFrame({
|
const frame2 = toDataFrame({
|
||||||
name: 'B',
|
name: 'B',
|
||||||
|
refId: 'B',
|
||||||
fields: [],
|
fields: [],
|
||||||
});
|
});
|
||||||
|
|
||||||
const frame3 = toDataFrame({
|
const frame3 = toDataFrame({
|
||||||
name: 'C',
|
name: 'C',
|
||||||
|
refId: 'C',
|
||||||
fields: [
|
fields: [
|
||||||
{ name: 'time', type: FieldType.time, values: [1, 2, 3] },
|
{ name: 'time', type: FieldType.time, values: [1, 2, 3] },
|
||||||
{ name: 'temperature', type: FieldType.number, values: [20, 22, 24] },
|
{ name: 'temperature', type: FieldType.number, values: [20, 22, 24] },
|
||||||
@ -1011,14 +1056,18 @@ describe('JOIN Transformer', () => {
|
|||||||
|
|
||||||
it('handles duplicate field name', async () => {
|
it('handles duplicate field name', async () => {
|
||||||
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
const cfg: DataTransformerConfig<JoinByFieldOptions> = {
|
||||||
id: DataTransformerID.seriesToColumns,
|
id: DataTransformerID.joinByField,
|
||||||
options: {
|
options: {
|
||||||
byField: 'time',
|
fields: {
|
||||||
|
frame1: 'time',
|
||||||
|
frame2: 'time',
|
||||||
|
},
|
||||||
mode: JoinMode.inner,
|
mode: JoinMode.inner,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
|
||||||
const frame1 = toDataFrame({
|
const frame1 = toDataFrame({
|
||||||
|
refId: 'frame1',
|
||||||
fields: [
|
fields: [
|
||||||
{ name: 'time', type: FieldType.time, values: [1] },
|
{ name: 'time', type: FieldType.time, values: [1] },
|
||||||
{ name: 'temperature', type: FieldType.number, values: [10] },
|
{ name: 'temperature', type: FieldType.number, values: [10] },
|
||||||
@ -1026,6 +1075,7 @@ describe('JOIN Transformer', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
const frame2 = toDataFrame({
|
const frame2 = toDataFrame({
|
||||||
|
refId: 'frame2',
|
||||||
fields: [
|
fields: [
|
||||||
{ name: 'time', type: FieldType.time, values: [1] },
|
{ name: 'time', type: FieldType.time, values: [1] },
|
||||||
{ name: 'temperature', type: FieldType.number, values: [20] },
|
{ name: 'temperature', type: FieldType.number, values: [20] },
|
||||||
|
@ -1,8 +1,6 @@
|
|||||||
import { map } from 'rxjs/operators';
|
import { map } from 'rxjs/operators';
|
||||||
|
|
||||||
import { DataFrame, SynchronousDataTransformerInfo, FieldMatcher } from '../../types';
|
import { DataFrame, SynchronousDataTransformerInfo } from '../../types';
|
||||||
import { fieldMatchers } from '../matchers';
|
|
||||||
import { FieldMatcherID } from '../matchers/ids';
|
|
||||||
|
|
||||||
import { DataTransformerID } from './ids';
|
import { DataTransformerID } from './ids';
|
||||||
import { joinDataFrames } from './joinDataFrames';
|
import { joinDataFrames } from './joinDataFrames';
|
||||||
@ -13,7 +11,7 @@ export enum JoinMode {
|
|||||||
}
|
}
|
||||||
|
|
||||||
export interface JoinByFieldOptions {
|
export interface JoinByFieldOptions {
|
||||||
byField?: string; // empty will pick the field automatically
|
fields?: { [key: string]: string }; // empty will pick the field automatically
|
||||||
mode?: JoinMode;
|
mode?: JoinMode;
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -24,7 +22,7 @@ export const joinByFieldTransformer: SynchronousDataTransformerInfo<JoinByFieldO
|
|||||||
description:
|
description:
|
||||||
'Combine rows from two or more tables, based on a related field between them. This can be used to outer join multiple time series on the _time_ field to show many time series in one table.',
|
'Combine rows from two or more tables, based on a related field between them. This can be used to outer join multiple time series on the _time_ field to show many time series in one table.',
|
||||||
defaultOptions: {
|
defaultOptions: {
|
||||||
byField: undefined, // DEFAULT_KEY_FIELD,
|
fields: {}, // DEFAULT_KEY_FIELD,
|
||||||
mode: JoinMode.outer,
|
mode: JoinMode.outer,
|
||||||
},
|
},
|
||||||
|
|
||||||
@ -32,17 +30,14 @@ export const joinByFieldTransformer: SynchronousDataTransformerInfo<JoinByFieldO
|
|||||||
source.pipe(map((data) => joinByFieldTransformer.transformer(options, ctx)(data))),
|
source.pipe(map((data) => joinByFieldTransformer.transformer(options, ctx)(data))),
|
||||||
|
|
||||||
transformer: (options: JoinByFieldOptions) => {
|
transformer: (options: JoinByFieldOptions) => {
|
||||||
let joinBy: FieldMatcher | undefined = undefined;
|
|
||||||
return (data: DataFrame[]) => {
|
return (data: DataFrame[]) => {
|
||||||
if (data.length > 1) {
|
if (data.length > 1) {
|
||||||
if (options.byField && !joinBy) {
|
const joined = joinDataFrames({ frames: data, mode: options.mode, fields: options.fields });
|
||||||
joinBy = fieldMatchers.get(FieldMatcherID.byName).get(options.byField);
|
|
||||||
}
|
|
||||||
const joined = joinDataFrames({ frames: data, joinBy, mode: options.mode });
|
|
||||||
if (joined) {
|
if (joined) {
|
||||||
return [joined];
|
return [joined];
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
return data;
|
return data;
|
||||||
};
|
};
|
||||||
},
|
},
|
||||||
|
@ -359,4 +359,103 @@ describe('align frames', () => {
|
|||||||
expect(isLikelyAscendingVector(new ArrayVector([null, 1, null]), 3)).toBeTruthy();
|
expect(isLikelyAscendingVector(new ArrayVector([null, 1, null]), 3)).toBeTruthy();
|
||||||
});
|
});
|
||||||
});
|
});
|
||||||
|
|
||||||
|
describe('should perform a join on custom fields', () => {
|
||||||
|
const tags = toDataFrame({
|
||||||
|
refId: 'tags',
|
||||||
|
fields: [
|
||||||
|
{ name: 'tags__time', type: FieldType.time, values: [100, 101, 200] },
|
||||||
|
{ name: 'tags__name', type: FieldType.string, values: ['v1.2', 'v1.2b', 'v1.3'] },
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
const releases = toDataFrame({
|
||||||
|
refId: 'releases',
|
||||||
|
fields: [
|
||||||
|
{ name: 'releases__time', type: FieldType.time, values: [150, 250] },
|
||||||
|
{ name: 'releases__tag', type: FieldType.string, values: ['v1.2', 'v1.3'] },
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
const features = toDataFrame({
|
||||||
|
refId: 'features',
|
||||||
|
fields: [
|
||||||
|
{ name: 'features__name', type: FieldType.string, values: ['A', 'B', 'C', 'D', 'E'] },
|
||||||
|
{ name: 'features__tag', type: FieldType.time, values: ['v1.2', 'v1.3', 'v1.2b', 'v1.3', 'v1.2'] },
|
||||||
|
],
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should perform an outer join', () => {
|
||||||
|
const out = joinDataFrames({
|
||||||
|
frames: [tags, releases, features],
|
||||||
|
fields: {
|
||||||
|
tags: 'tags__name',
|
||||||
|
releases: 'releases__tag',
|
||||||
|
features: 'features__tag',
|
||||||
|
},
|
||||||
|
})!;
|
||||||
|
|
||||||
|
expect(
|
||||||
|
out.fields.map((f) => ({
|
||||||
|
name: f.name,
|
||||||
|
values: f.values.toArray(),
|
||||||
|
}))
|
||||||
|
).toEqual([
|
||||||
|
{
|
||||||
|
name: 'tags__name',
|
||||||
|
values: ['v1.2', 'v1.2b', 'v1.3'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'tags__time',
|
||||||
|
values: [100, 101, 200],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'releases__time',
|
||||||
|
values: [150, undefined, 250],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'features__name',
|
||||||
|
values: ['E', 'C', 'D'],
|
||||||
|
},
|
||||||
|
]);
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should perform an inner join', () => {
|
||||||
|
const out = joinDataFrames({
|
||||||
|
frames: [tags, releases, features],
|
||||||
|
fields: {
|
||||||
|
tags: 'tags__name',
|
||||||
|
releases: 'releases__tag',
|
||||||
|
features: 'features__tag',
|
||||||
|
},
|
||||||
|
mode: JoinMode.inner,
|
||||||
|
})!;
|
||||||
|
|
||||||
|
const mappedOut = out.fields.map((f) => ({
|
||||||
|
name: f.name,
|
||||||
|
values: f.values.toArray(),
|
||||||
|
}));
|
||||||
|
|
||||||
|
const expected = [
|
||||||
|
{
|
||||||
|
name: 'tags__name',
|
||||||
|
values: ['v1.2', 'v1.3'],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'tags__time',
|
||||||
|
values: [100, 200],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'releases__time',
|
||||||
|
values: [150, 250],
|
||||||
|
},
|
||||||
|
{
|
||||||
|
name: 'features__name',
|
||||||
|
values: ['E', 'D'],
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
expect(JSON.stringify(mappedOut)).toEqual(JSON.stringify(expected));
|
||||||
|
});
|
||||||
|
});
|
||||||
});
|
});
|
||||||
|
@ -47,6 +47,11 @@ export interface JoinOptions {
|
|||||||
*/
|
*/
|
||||||
joinBy?: FieldMatcher;
|
joinBy?: FieldMatcher;
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The fields to join on
|
||||||
|
*/
|
||||||
|
fields?: { [key: string]: string };
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Optionally filter the non-join fields
|
* Optionally filter the non-join fields
|
||||||
*/
|
*/
|
||||||
@ -63,8 +68,16 @@ export interface JoinOptions {
|
|||||||
mode?: JoinMode;
|
mode?: JoinMode;
|
||||||
}
|
}
|
||||||
|
|
||||||
function getJoinMatcher(options: JoinOptions): FieldMatcher {
|
function getJoinMatcher(options: JoinOptions, refId: string | undefined): FieldMatcher {
|
||||||
return options.joinBy ?? pickBestJoinField(options.frames);
|
if (options.joinBy) {
|
||||||
|
return options.joinBy;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!options.fields || !refId) {
|
||||||
|
return pickBestJoinField(options.frames);
|
||||||
|
}
|
||||||
|
|
||||||
|
return fieldMatchers.get(FieldMatcherID.byName).get(options.fields[refId]);
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -95,7 +108,7 @@ export function joinDataFrames(options: JoinOptions): DataFrame | undefined {
|
|||||||
let frame = options.frames[0];
|
let frame = options.frames[0];
|
||||||
let frameCopy = frame;
|
let frameCopy = frame;
|
||||||
|
|
||||||
const joinFieldMatcher = getJoinMatcher(options);
|
const joinFieldMatcher = getJoinMatcher(options, frame.refId);
|
||||||
let joinIndex = frameCopy.fields.findIndex((f) => joinFieldMatcher(f, frameCopy, options.frames));
|
let joinIndex = frameCopy.fields.findIndex((f) => joinFieldMatcher(f, frameCopy, options.frames));
|
||||||
|
|
||||||
if (options.keepOriginIndices) {
|
if (options.keepOriginIndices) {
|
||||||
@ -152,10 +165,10 @@ export function joinDataFrames(options: JoinOptions): DataFrame | undefined {
|
|||||||
const nullModes: JoinNullMode[][] = [];
|
const nullModes: JoinNullMode[][] = [];
|
||||||
const allData: AlignedData[] = [];
|
const allData: AlignedData[] = [];
|
||||||
const originalFields: Field[] = [];
|
const originalFields: Field[] = [];
|
||||||
const joinFieldMatcher = getJoinMatcher(options);
|
|
||||||
|
|
||||||
for (let frameIndex = 0; frameIndex < options.frames.length; frameIndex++) {
|
for (let frameIndex = 0; frameIndex < options.frames.length; frameIndex++) {
|
||||||
const frame = options.frames[frameIndex];
|
const frame = options.frames[frameIndex];
|
||||||
|
const joinFieldMatcher = getJoinMatcher(options, frame.refId);
|
||||||
|
|
||||||
if (!frame || !frame.fields?.length) {
|
if (!frame || !frame.fields?.length) {
|
||||||
continue; // skip the frame
|
continue; // skip the frame
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
import React, { useCallback } from 'react';
|
import React, { useCallback, useEffect } from 'react';
|
||||||
|
|
||||||
import {
|
import {
|
||||||
DataTransformerID,
|
DataTransformerID,
|
||||||
@ -6,11 +6,10 @@ import {
|
|||||||
standardTransformers,
|
standardTransformers,
|
||||||
TransformerRegistryItem,
|
TransformerRegistryItem,
|
||||||
TransformerUIProps,
|
TransformerUIProps,
|
||||||
|
DataFrame,
|
||||||
} from '@grafana/data';
|
} from '@grafana/data';
|
||||||
import { JoinByFieldOptions, JoinMode } from '@grafana/data/src/transformations/transformers/joinByField';
|
import { JoinByFieldOptions, JoinMode } from '@grafana/data/src/transformations/transformers/joinByField';
|
||||||
import { Select, InlineFieldRow, InlineField } from '@grafana/ui';
|
import { Select, InlineFieldRow, InlineField, Checkbox, HorizontalGroup } from '@grafana/ui';
|
||||||
|
|
||||||
import { useAllFieldNamesFromDataFrames } from '../utils';
|
|
||||||
|
|
||||||
const modes = [
|
const modes = [
|
||||||
{ value: JoinMode.outer, label: 'OUTER', description: 'Keep all rows from any table with a value' },
|
{ value: JoinMode.outer, label: 'OUTER', description: 'Keep all rows from any table with a value' },
|
||||||
@ -18,14 +17,44 @@ const modes = [
|
|||||||
];
|
];
|
||||||
|
|
||||||
export function SeriesToFieldsTransformerEditor({ input, options, onChange }: TransformerUIProps<JoinByFieldOptions>) {
|
export function SeriesToFieldsTransformerEditor({ input, options, onChange }: TransformerUIProps<JoinByFieldOptions>) {
|
||||||
const fieldNames = useAllFieldNamesFromDataFrames(input).map((item: string) => ({ label: item, value: item }));
|
useEffect(() => {
|
||||||
|
if (options.fields && !Object.keys(options.fields).length && input.length && input[0].refId) {
|
||||||
|
options.fields[input[0].refId] = input[0].fields[0].name;
|
||||||
|
onChange({ ...options });
|
||||||
|
}
|
||||||
|
}, [onChange, options, input]);
|
||||||
|
|
||||||
|
const onToggleDataFrame = useCallback(
|
||||||
|
(dataFrame: DataFrame) => {
|
||||||
|
if (!dataFrame.refId) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (options.fields) {
|
||||||
|
if (dataFrame.refId in options.fields) {
|
||||||
|
if (Object.keys(options.fields).length === 1) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
delete options.fields[dataFrame.refId];
|
||||||
|
} else {
|
||||||
|
options.fields[dataFrame.refId] = dataFrame.fields[0].name;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onChange({ ...options });
|
||||||
|
},
|
||||||
|
[onChange, options]
|
||||||
|
);
|
||||||
|
|
||||||
const onSelectField = useCallback(
|
const onSelectField = useCallback(
|
||||||
(value: SelectableValue<string>) => {
|
(queryRefId: string | undefined, fieldName: SelectableValue<string>) => {
|
||||||
onChange({
|
if (queryRefId && fieldName.value) {
|
||||||
...options,
|
onChange({
|
||||||
byField: value?.value,
|
...options,
|
||||||
});
|
fields: { ...options.fields, [queryRefId]: fieldName.value },
|
||||||
|
});
|
||||||
|
}
|
||||||
},
|
},
|
||||||
[onChange, options]
|
[onChange, options]
|
||||||
);
|
);
|
||||||
@ -34,7 +63,7 @@ export function SeriesToFieldsTransformerEditor({ input, options, onChange }: Tr
|
|||||||
(value: SelectableValue<JoinMode>) => {
|
(value: SelectableValue<JoinMode>) => {
|
||||||
onChange({
|
onChange({
|
||||||
...options,
|
...options,
|
||||||
mode: value?.value,
|
mode: value?.value || JoinMode.outer,
|
||||||
});
|
});
|
||||||
},
|
},
|
||||||
[onChange, options]
|
[onChange, options]
|
||||||
@ -44,20 +73,31 @@ export function SeriesToFieldsTransformerEditor({ input, options, onChange }: Tr
|
|||||||
<>
|
<>
|
||||||
<InlineFieldRow>
|
<InlineFieldRow>
|
||||||
<InlineField label="Mode" labelWidth={8} grow>
|
<InlineField label="Mode" labelWidth={8} grow>
|
||||||
<Select options={modes} value={options.mode ?? JoinMode.outer} onChange={onSetMode} />
|
<Select options={modes} value={options.mode} onChange={onSetMode} />
|
||||||
</InlineField>
|
|
||||||
</InlineFieldRow>
|
|
||||||
<InlineFieldRow>
|
|
||||||
<InlineField label="Field" labelWidth={8} grow>
|
|
||||||
<Select
|
|
||||||
options={fieldNames}
|
|
||||||
value={options.byField}
|
|
||||||
onChange={onSelectField}
|
|
||||||
placeholder="time"
|
|
||||||
isClearable
|
|
||||||
/>
|
|
||||||
</InlineField>
|
</InlineField>
|
||||||
</InlineFieldRow>
|
</InlineFieldRow>
|
||||||
|
{input.map((dataFrame) => (
|
||||||
|
<div className="gf-form-inline" key={dataFrame.refId}>
|
||||||
|
<div className="gf-form gf-form--grow">
|
||||||
|
<div className="gf-form-label width-8">
|
||||||
|
{dataFrame.refId} ({dataFrame.name})
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<HorizontalGroup>
|
||||||
|
<Checkbox
|
||||||
|
value={!!dataFrame.refId && options.fields && dataFrame.refId in options.fields}
|
||||||
|
onChange={() => onToggleDataFrame(dataFrame)}
|
||||||
|
/>
|
||||||
|
|
||||||
|
<Select
|
||||||
|
options={dataFrame.fields.map((field) => ({ label: field.name, value: field.name }))}
|
||||||
|
value={dataFrame.refId ? (options.fields || {})[dataFrame.refId] : dataFrame.fields[0].name}
|
||||||
|
onChange={(fieldName) => onSelectField(dataFrame.refId, fieldName)}
|
||||||
|
/>
|
||||||
|
</HorizontalGroup>
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
))}
|
||||||
</>
|
</>
|
||||||
);
|
);
|
||||||
}
|
}
|
||||||
|
Loading…
Reference in New Issue
Block a user