diff --git a/circle.yml b/.circleci/config.yml
similarity index 100%
rename from circle.yml
rename to .circleci/config.yml
diff --git a/CHANGELOG.md b/CHANGELOG.md
index 304b1ba6d0b..1df6266c763 100644
--- a/CHANGELOG.md
+++ b/CHANGELOG.md
@@ -3,12 +3,14 @@
* **MSSQL**: New Microsoft SQL Server data source [#10093](https://github.com/grafana/grafana/pull/10093), [#11298](https://github.com/grafana/grafana/pull/11298), thx [@linuxchips](https://github.com/linuxchips)
* **Prometheus**: The heatmap panel now support Prometheus histograms [#10009](https://github.com/grafana/grafana/issues/10009)
* **Postgres/MySQL**: Ability to insert 0s or nulls for missing intervals [#9487](https://github.com/grafana/grafana/issues/9487), thanks [@svenklemm](https://github.com/svenklemm)
+* **Graph**: Align left and right Y-axes to one level [#1271](https://github.com/grafana/grafana/issues/1271) & [#2740](https://github.com/grafana/grafana/issues/2740) thx [@ilgizar](https://github.com/ilgizar)
* **Graph**: Thresholds for Right Y axis [#7107](https://github.com/grafana/grafana/issues/7107), thx [@ilgizar](https://github.com/ilgizar)
* **Graph**: Support multiple series stacking in histogram mode [#8151](https://github.com/grafana/grafana/issues/8151), thx [@mtanda](https://github.com/mtanda)
* **Alerting**: Pausing/un alerts now updates new_state_date [#10942](https://github.com/grafana/grafana/pull/10942)
* **Alerting**: Support Pagerduty notification channel using Pagerduty V2 API [#10531](https://github.com/grafana/grafana/issues/10531), thx [@jbaublitz](https://github.com/jbaublitz)
* **Templating**: Add comma templating format [#10632](https://github.com/grafana/grafana/issues/10632), thx [@mtanda](https://github.com/mtanda)
* **Prometheus**: Support POST for query and query_range [#9859](https://github.com/grafana/grafana/pull/9859), thx [@mtanda](https://github.com/mtanda)
+* **Alerting**: Add support for retries on alert queries [#5855](https://github.com/grafana/grafana/issues/5855), thx [@Thib17](https://github.com/Thib17)
### Minor
* **OpsGenie**: Add triggered alerts as description [#11046](https://github.com/grafana/grafana/pull/11046), thx [@llamashoes](https://github.com/llamashoes)
@@ -16,6 +18,7 @@
* **Cloudwatch**: Add dimension filtering to CloudWatch `dimension_values()` [#10029](https://github.com/grafana/grafana/issues/10029), thx [@willyhutw](https://github.com/willyhutw)
* **Units**: Second to HH:mm:ss formatter [#11107](https://github.com/grafana/grafana/issues/11107), thx [@gladdiologist](https://github.com/gladdiologist)
* **Singlestat**: Add color to prefix and postfix in singlestat panel [#11143](https://github.com/grafana/grafana/pull/11143), thx [@ApsOps](https://github.com/ApsOps)
+* **Dashboards**: Version cleanup fails on old databases with many entries [#11278](https://github.com/grafana/grafana/issues/11278)
# 5.0.4 (unreleased)
* **Dashboard** Fixed bug where collapsed panels could not be directly linked to/renderer [#11114](https://github.com/grafana/grafana/issues/11114) & [#11086](https://github.com/grafana/grafana/issues/11086)
diff --git a/docker/blocks/mssql_tests/dashboard.json b/docker/blocks/mssql_tests/dashboard.json
index 323a61bb49a..20e3907b48b 100644
--- a/docker/blocks/mssql_tests/dashboard.json
+++ b/docker/blocks/mssql_tests/dashboard.json
@@ -53,7 +53,7 @@
"iconColor": "#6ed0e0",
"limit": 100,
"name": "Deploys",
- "rawQuery": "SELECT\n time_sec as time,\n description as [text],\n tags\n FROM [event]\n WHERE $__unixEpochFilter(time_sec) AND tags='deploy'\n ORDER BY 1 ASC\n ",
+ "rawQuery": "SELECT\n $__time(time_sec),\n description as [text],\n tags\n FROM [event]\n WHERE $__unixEpochFilter(time_sec) AND tags='deploy'\n ORDER BY 1 ASC\n ",
"showIn": 0,
"tags": [],
"type": "tags"
@@ -65,7 +65,7 @@
"iconColor": "rgba(255, 96, 96, 1)",
"limit": 100,
"name": "Tickets",
- "rawQuery": "SELECT\n time_sec as time,\n description as [text],\n tags\n FROM [event]\n WHERE $__unixEpochFilter(time_sec) AND tags='ticket'\n ORDER BY 1 ASC\n ",
+ "rawQuery": "SELECT\n $__time(time_sec),\n description as [text],\n tags\n FROM [event]\n WHERE $__unixEpochFilter(time_sec) AND tags='ticket'\n ORDER BY 1 ASC\n ",
"showIn": 0,
"tags": [],
"type": "tags"
@@ -76,8 +76,20 @@
"hide": false,
"iconColor": "#7eb26d",
"limit": 100,
- "name": "Metric Values",
- "rawQuery": "SELECT \n time, \n measurement as text, \n '' as tags\nFROM\n metric_values \nORDER BY 1",
+ "name": "Metric Values timeEpoch macro",
+ "rawQuery": "SELECT \n $__timeEpoch(time), \n measurement as text, \n '' as tags\nFROM\n metric_values \nWHERE\n $__timeFilter(time)\nORDER BY 1",
+ "showIn": 0,
+ "tags": [],
+ "type": "tags"
+ },
+ {
+ "datasource": "${DS_MSSQL_TEST}",
+ "enable": false,
+ "hide": false,
+ "iconColor": "#1f78c1",
+ "limit": 100,
+ "name": "Metric Values native time",
+ "rawQuery": "SELECT \n time, \n measurement as text, \n '' as tags\nFROM\n metric_values \nWHERE\n $__timeFilter(time)\nORDER BY 1",
"showIn": 0,
"tags": [],
"type": "tags"
@@ -88,7 +100,7 @@
"gnetId": null,
"graphTooltip": 0,
"id": null,
- "iteration": 1521481503341,
+ "iteration": 1521715844826,
"links": [],
"panels": [
{
@@ -138,6 +150,222 @@
"transform": "table",
"type": "table"
},
+ {
+ "columns": [],
+ "datasource": "${DS_MSSQL_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 0,
+ "y": 4
+ },
+ "id": 32,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT cast(null as bigint) as time",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "cast(null as bigint) as time",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "columns": [],
+ "datasource": "${DS_MSSQL_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 6,
+ "y": 4
+ },
+ "id": 33,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT cast(null as datetime) as time",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "cast(null as datetime) as time",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "columns": [],
+ "datasource": "${DS_MSSQL_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 12,
+ "y": 4
+ },
+ "id": 34,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT GETDATE() as time",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "GETDATE() as time",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "columns": [],
+ "datasource": "${DS_MSSQL_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 18,
+ "y": 4
+ },
+ "id": 35,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT GETUTCDATE() as time",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "GETUTCDATE() as time",
+ "transform": "table",
+ "type": "table"
+ },
{
"aliasColors": {},
"bars": false,
@@ -149,7 +377,7 @@
"h": 9,
"w": 8,
"x": 0,
- "y": 4
+ "y": 7
},
"id": 7,
"legend": {
@@ -228,7 +456,7 @@
"h": 9,
"w": 8,
"x": 8,
- "y": 4
+ "y": 7
},
"id": 9,
"legend": {
@@ -307,7 +535,7 @@
"h": 9,
"w": 8,
"x": 16,
- "y": 4
+ "y": 7
},
"id": 10,
"legend": {
@@ -386,7 +614,7 @@
"h": 9,
"w": 8,
"x": 0,
- "y": 13
+ "y": 16
},
"id": 16,
"legend": {
@@ -465,7 +693,7 @@
"h": 9,
"w": 8,
"x": 8,
- "y": 13
+ "y": 16
},
"id": 12,
"legend": {
@@ -544,7 +772,7 @@
"h": 9,
"w": 8,
"x": 16,
- "y": 13
+ "y": 16
},
"id": 13,
"legend": {
@@ -623,7 +851,7 @@
"h": 8,
"w": 12,
"x": 0,
- "y": 22
+ "y": 25
},
"id": 27,
"legend": {
@@ -655,13 +883,13 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT \n $__timeGroup(time, '$summarize') as time, \n measurement + ' - value one' as metric, \n avg(valueOne) as valueOne\nFROM\n metric_values \nWHERE\n $__timeFilter(time)\nGROUP BY \n $__timeGroup(time, '$summarize'), \n measurement \nORDER BY 1",
+ "rawSql": "SELECT \n $__timeGroup(time, '$summarize') as time, \n measurement + ' - value one' as metric, \n avg(valueOne) as valueOne\nFROM\n metric_values \nWHERE\n $__timeFilter(time) AND\n ($metric = 'ALL' OR measurement = $metric)\nGROUP BY \n $__timeGroup(time, '$summarize'), \n measurement \nORDER BY 1",
"refId": "A"
},
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT \n $__timeGroup(time, '$summarize') as time, \n measurement + ' - value two' as metric, \n avg(valueTwo) as valueTwo \nFROM\n metric_values \nGROUP BY \n $__timeGroup(time, '$summarize'), \n measurement \nORDER BY 1",
+ "rawSql": "SELECT \n $__timeGroup(time, '$summarize') as time, \n measurement + ' - value two' as metric, \n avg(valueTwo) as valueTwo \nFROM\n metric_values\nWHERE\n $__timeFilter(time) AND\n ($metric = 'ALL' OR measurement = $metric)\nGROUP BY \n $__timeGroup(time, '$summarize'), \n measurement \nORDER BY 1",
"refId": "B"
}
],
@@ -712,7 +940,7 @@
"h": 8,
"w": 12,
"x": 12,
- "y": 22
+ "y": 25
},
"id": 5,
"legend": {
@@ -734,7 +962,19 @@
"pointradius": 3,
"points": false,
"renderer": "flot",
- "seriesOverrides": [],
+ "seriesOverrides": [
+ {
+ "alias": "MovingAverageValueOne",
+ "dashes": true,
+ "lines": false
+ },
+ {
+ "alias": "MovingAverageValueTwo",
+ "dashes": true,
+ "lines": false,
+ "yaxis": 1
+ }
+ ],
"spaceLength": 10,
"stack": false,
"steppedLine": false,
@@ -742,8 +982,14 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT \n $__timeGroup(time, '$summarize') as time, \n avg(valueOne) as valueOne, \n avg(valueTwo) as valueTwo \nFROM\n metric_values \nGROUP BY \n $__timeGroup(time, '$summarize')\nORDER BY 1",
+ "rawSql": "SELECT \n $__timeGroup(time, '$summarize') as time, \n avg(valueOne) as valueOne, \n avg(valueTwo) as valueTwo \nFROM\n metric_values \nWHERE \n $__timeFilter(time) AND \n ($metric = 'ALL' OR measurement = $metric)\nGROUP BY \n $__timeGroup(time, '$summarize')\nORDER BY 1",
"refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT \n time,\n avg(valueOne) OVER (ORDER BY time ROWS BETWEEN 6 PRECEDING AND 6 FOLLOWING) as MovingAverageValueOne,\n avg(valueTwo) OVER (ORDER BY time ROWS BETWEEN 6 PRECEDING AND 6 FOLLOWING) as MovingAverageValueTwo\nFROM\n metric_values \nWHERE \n $__timeFilter(time) AND \n ($metric = 'ALL' OR measurement = $metric)\nORDER BY 1",
+ "refId": "B"
}
],
"thresholds": [],
@@ -793,7 +1039,7 @@
"h": 8,
"w": 12,
"x": 0,
- "y": 30
+ "y": 33
},
"id": 4,
"legend": {
@@ -825,13 +1071,13 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values WHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
},
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values WHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "B"
}
],
@@ -882,7 +1128,7 @@
"h": 8,
"w": 12,
"x": 12,
- "y": 30
+ "y": 33
},
"id": 28,
"legend": {
@@ -963,7 +1209,7 @@
"h": 8,
"w": 12,
"x": 0,
- "y": 38
+ "y": 41
},
"id": 19,
"legend": {
@@ -995,13 +1241,13 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values WHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
},
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values WHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "B"
}
],
@@ -1052,7 +1298,7 @@
"h": 8,
"w": 12,
"x": 12,
- "y": 38
+ "y": 41
},
"id": 18,
"legend": {
@@ -1082,7 +1328,7 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values\nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
}
],
@@ -1133,7 +1379,7 @@
"h": 8,
"w": 12,
"x": 0,
- "y": 46
+ "y": 49
},
"id": 17,
"legend": {
@@ -1165,13 +1411,13 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values WHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
},
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values WHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "B"
}
],
@@ -1222,7 +1468,7 @@
"h": 8,
"w": 12,
"x": 12,
- "y": 46
+ "y": 49
},
"id": 20,
"legend": {
@@ -1252,7 +1498,7 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values\nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
}
],
@@ -1303,7 +1549,7 @@
"h": 8,
"w": 12,
"x": 0,
- "y": 54
+ "y": 57
},
"id": 29,
"legend": {
@@ -1335,7 +1581,7 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "DECLARE \n @from int = $__unixEpochFrom(),\n @to int = $__unixEpochTo()\n \nEXEC dbo.sp_test_epoch @from, @to",
+ "rawSql": "DECLARE\n @from int = $__unixEpochFrom(), \n @to int = $__unixEpochTo(), \n @interval nvarchar(50) = '$summarize', \n @metric nvarchar(200) = $metric\n \nEXEC dbo.sp_test_epoch @from, @to, @interval, @metric",
"refId": "A"
}
],
@@ -1386,7 +1632,7 @@
"h": 8,
"w": 12,
"x": 12,
- "y": 54
+ "y": 57
},
"id": 30,
"legend": {
@@ -1418,7 +1664,7 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "DECLARE \n @from datetime = $__timeFrom(),\n @to datetime = $__timeTo()\n \nEXEC dbo.sp_test_datetime @from, @to",
+ "rawSql": "DECLARE\n @from datetime = $__timeFrom(), \n @to datetime = $__timeTo(), \n @interval nvarchar(50) = '$summarize', \n @metric nvarchar(200) = $metric\n \nEXEC dbo.sp_test_datetime @from, @to, @interval, @metric",
"refId": "A"
}
],
@@ -1469,7 +1715,7 @@
"h": 8,
"w": 12,
"x": 0,
- "y": 62
+ "y": 65
},
"id": 14,
"legend": {
@@ -1499,13 +1745,13 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values \nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
},
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values \nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "B"
}
],
@@ -1559,7 +1805,7 @@
"h": 8,
"w": 12,
"x": 12,
- "y": 62
+ "y": 65
},
"id": 15,
"legend": {
@@ -1589,7 +1835,7 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values\nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
}
],
@@ -1642,7 +1888,7 @@
"h": 8,
"w": 12,
"x": 0,
- "y": 70
+ "y": 73
},
"id": 25,
"legend": {
@@ -1672,13 +1918,13 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values \nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
},
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values \nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "B"
}
],
@@ -1732,7 +1978,7 @@
"h": 8,
"w": 12,
"x": 12,
- "y": 70
+ "y": 73
},
"id": 22,
"legend": {
@@ -1762,7 +2008,7 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values\nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
}
],
@@ -1815,7 +2061,7 @@
"h": 8,
"w": 12,
"x": 0,
- "y": 78
+ "y": 81
},
"id": 21,
"legend": {
@@ -1845,13 +2091,13 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values \nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
},
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values \nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "B"
}
],
@@ -1905,7 +2151,7 @@
"h": 8,
"w": 12,
"x": 12,
- "y": 78
+ "y": 81
},
"id": 26,
"legend": {
@@ -1935,7 +2181,7 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values \nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
}
],
@@ -1988,7 +2234,7 @@
"h": 8,
"w": 12,
"x": 0,
- "y": 86
+ "y": 89
},
"id": 23,
"legend": {
@@ -2018,13 +2264,13 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value one' as metric, valueOne FROM metric_values\nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
},
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), measurement + ' - value two' as metric, valueTwo FROM metric_values \nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "B"
}
],
@@ -2078,7 +2324,7 @@
"h": 8,
"w": 12,
"x": 12,
- "y": 86
+ "y": 89
},
"id": 24,
"legend": {
@@ -2108,7 +2354,7 @@
{
"alias": "",
"format": "time_series",
- "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values ORDER BY 1",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values \nWHERE $__timeFilter(time) AND ($metric = 'ALL' OR measurement = $metric) ORDER BY 1",
"refId": "A"
}
],
@@ -2157,6 +2403,26 @@
"tags": [],
"templating": {
"list": [
+ {
+ "allValue": "'ALL'",
+ "current": {},
+ "datasource": "${DS_MSSQL_TEST}",
+ "hide": 0,
+ "includeAll": true,
+ "label": "Metric",
+ "multi": false,
+ "name": "metric",
+ "options": [],
+ "query": "SELECT DISTINCT measurement FROM metric_values",
+ "refresh": 1,
+ "regex": "",
+ "sort": 0,
+ "tagValuesQuery": "",
+ "tags": [],
+ "tagsQuery": "",
+ "type": "query",
+ "useTags": false
+ },
{
"auto": false,
"auto_count": 30,
@@ -2208,7 +2474,7 @@
},
"time": {
"from": "2018-03-15T12:30:00.000Z",
- "to": "2018-03-15T13:55:00.000Z"
+ "to": "2018-03-15T13:55:01.000Z"
},
"timepicker": {
"refresh_intervals": [
@@ -2238,5 +2504,5 @@
"timezone": "",
"title": "Microsoft SQL Server Data Source Test",
"uid": "GlAqcPgmz",
- "version": 37
+ "version": 57
}
\ No newline at end of file
diff --git a/docker/blocks/mysql_tests/dashboard.json b/docker/blocks/mysql_tests/dashboard.json
new file mode 100644
index 00000000000..3ab08a7da35
--- /dev/null
+++ b/docker/blocks/mysql_tests/dashboard.json
@@ -0,0 +1,2350 @@
+{
+ "__inputs": [
+ {
+ "name": "DS_MYSQL_TEST",
+ "label": "MySQL TEST",
+ "description": "",
+ "type": "datasource",
+ "pluginId": "mysql",
+ "pluginName": "MySQL"
+ },
+ {
+ "name": "DS_MSSQL_TEST",
+ "label": "MSSQL Test",
+ "description": "",
+ "type": "datasource",
+ "pluginId": "mssql",
+ "pluginName": "Microsoft SQL Server"
+ }
+ ],
+ "__requires": [
+ {
+ "type": "grafana",
+ "id": "grafana",
+ "name": "Grafana",
+ "version": "5.0.0"
+ },
+ {
+ "type": "panel",
+ "id": "graph",
+ "name": "Graph",
+ "version": "5.0.0"
+ },
+ {
+ "type": "datasource",
+ "id": "mssql",
+ "name": "Microsoft SQL Server",
+ "version": "1.0.0"
+ },
+ {
+ "type": "datasource",
+ "id": "mysql",
+ "name": "MySQL",
+ "version": "5.0.0"
+ },
+ {
+ "type": "panel",
+ "id": "table",
+ "name": "Table",
+ "version": "5.0.0"
+ }
+ ],
+ "annotations": {
+ "list": [
+ {
+ "builtIn": 1,
+ "datasource": "-- Grafana --",
+ "enable": true,
+ "hide": true,
+ "iconColor": "rgba(0, 211, 255, 1)",
+ "name": "Annotations & Alerts",
+ "type": "dashboard"
+ },
+ {
+ "datasource": "${DS_MYSQL_TEST}",
+ "enable": false,
+ "hide": false,
+ "iconColor": "#6ed0e0",
+ "limit": 100,
+ "name": "Deploys",
+ "rawQuery": "SELECT\n time_sec,\n description as text,\n tags\n FROM event\n WHERE $__unixEpochFilter(time_sec) AND tags='deploy'\n ORDER BY 1 ASC\n ",
+ "showIn": 0,
+ "tags": [],
+ "type": "tags"
+ },
+ {
+ "datasource": "${DS_MYSQL_TEST}",
+ "enable": false,
+ "hide": false,
+ "iconColor": "rgba(255, 96, 96, 1)",
+ "limit": 100,
+ "name": "Tickets",
+ "rawQuery": "SELECT\n time_sec as time,\n description as text,\n tags\n FROM event\n WHERE $__unixEpochFilter(time_sec) AND tags='ticket'\n ORDER BY 1 ASC\n ",
+ "showIn": 0,
+ "tags": [],
+ "type": "tags"
+ },
+ {
+ "datasource": "${DS_MYSQL_TEST}",
+ "enable": false,
+ "hide": false,
+ "iconColor": "#7eb26d",
+ "limit": 100,
+ "name": "Metric Values timeEpoch macro",
+ "rawQuery": "SELECT \n $__timeEpoch(time), \n measurement as text, \n '' as tags\nFROM\n metric_values \nWHERE\n $__timeFilter(time)\nORDER BY 1",
+ "showIn": 0,
+ "tags": [],
+ "type": "tags"
+ },
+ {
+ "datasource": "${DS_MYSQL_TEST}",
+ "enable": false,
+ "hide": false,
+ "iconColor": "#1f78c1",
+ "limit": 100,
+ "name": "Metric Values native time",
+ "rawQuery": "SELECT \n time, \n measurement as text, \n '' as tags\nFROM\n metric_values \nWHERE\n $__timeFilter(time)\nORDER BY 1",
+ "showIn": 0,
+ "tags": [],
+ "type": "tags"
+ }
+ ]
+ },
+ "editable": true,
+ "gnetId": null,
+ "graphTooltip": 0,
+ "id": null,
+ "iteration": 1521715720483,
+ "links": [],
+ "panels": [
+ {
+ "columns": [],
+ "datasource": "${DS_MYSQL_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 4,
+ "w": 24,
+ "x": 0,
+ "y": 0
+ },
+ "id": 2,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "string",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT * from mysql_types",
+ "refId": "A"
+ }
+ ],
+ "title": "Data types",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "columns": [],
+ "datasource": "${DS_MYSQL_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 0,
+ "y": 4
+ },
+ "id": 32,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time_sec",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT cast(null as unsigned integer) as time_sec",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "cast(null as unsigned integer) as time",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "columns": [],
+ "datasource": "${DS_MYSQL_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 6,
+ "y": 4
+ },
+ "id": 33,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time_sec",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT cast(null as datetime) as time_sec",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "cast(null as datetime) as time",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "columns": [],
+ "datasource": "${DS_MYSQL_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 12,
+ "y": 4
+ },
+ "id": 34,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time_sec",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT cast(NOW() as datetime) as time_sec",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "cast()NOW() as datetime) as time",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "columns": [],
+ "datasource": "${DS_MYSQL_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 18,
+ "y": 4
+ },
+ "id": 35,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time_sec",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT NOW() as time",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "NOW() as time",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 0,
+ "y": 7
+ },
+ "id": 7,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": true,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '5m') AS time, avg(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "timeGroup macro 5m without fill",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 8,
+ "y": 7
+ },
+ "id": 9,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null as zero",
+ "percentage": false,
+ "pointradius": 3,
+ "points": true,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '5m', NULL) AS time, avg(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "timeGroup macro 5m with fill(NULL) and null as zero",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 16,
+ "y": 7
+ },
+ "id": 10,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": true,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '5m', 10.0) AS time, avg(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "timeGroup macro 5m with fill(10.0)",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 0,
+ "y": 16
+ },
+ "id": 16,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": false,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '$summarize') AS time, avg(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Metrics - timeGroup macro $summarize without fill",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 8,
+ "y": 16
+ },
+ "id": 12,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": false,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null as zero",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '$summarize', NULL) AS time, sum(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Metrics - timeGroup macro $summarize with fill(NULL)",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 16,
+ "y": 16
+ },
+ "id": 13,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": false,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '$summarize', 100.0) AS time, sum(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Metrics - timeGroup macro $summarize with fill(100.0)",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 25
+ },
+ "id": 27,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "hideEmpty": false,
+ "hideZero": false,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT \n $__timeGroup(time, '$summarize') as time, \n CONCAT(measurement, ' - value one') as metric, \n avg(valueOne) as valueOne\nFROM\n metric_values \nWHERE\n $__timeFilter(time) AND\n measurement IN($metric)\nGROUP BY 1, 2\nORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT \n $__timeGroup(time, '$summarize') as time, \n CONCAT(measurement, ' - value two') as metric, \n avg(valueTwo) as valueTwo \nFROM\n metric_values\nWHERE\n $__timeFilter(time) AND\n measurement IN($metric)\nGROUP BY 1,2\nORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column using timeGroup macro ($summarize)",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 25
+ },
+ "id": 5,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [
+ {
+ "alias": "MovingAverageValueOne",
+ "dashes": true,
+ "lines": false
+ },
+ {
+ "alias": "MovingAverageValueTwo",
+ "dashes": true,
+ "lines": false,
+ "yaxis": 1
+ }
+ ],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT \n $__timeGroup(time, '$summarize') as time, \n avg(valueOne) as valueOne, \n avg(valueTwo) as valueTwo \nFROM\n metric_values \nWHERE \n $__timeFilter(time) AND \n measurement IN($metric)\nGROUP BY 1\nORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column using timeGroup macro ($summarize)",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 33
+ },
+ "id": 4,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "hideEmpty": false,
+ "hideZero": false,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__time(time), CONCAT(measurement, ' - value one') as metric, valueOne FROM metric_values WHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__time(time), CONCAT(measurement, ' - value two') as metric, valueTwo FROM metric_values WHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 33
+ },
+ "id": 28,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__time(time), valueOne, valueTwo FROM metric_values ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 41
+ },
+ "id": 19,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "hideEmpty": false,
+ "hideZero": false,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__time(time), CONCAT(measurement, ' - value one') as metric, valueOne FROM metric_values WHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__time(time), CONCAT(measurement, ' - value two') as metric, valueTwo FROM metric_values WHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - stacked",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 41
+ },
+ "id": 18,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values\nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - stacked",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 49
+ },
+ "id": 17,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "hideEmpty": false,
+ "hideZero": false,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": true,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__time(time), CONCAT(measurement, ' - value one') as metric, valueOne FROM metric_values WHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__time(time), CONCAT(measurement, ' - value two') as metric, valueTwo FROM metric_values WHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - stacked percent",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 49
+ },
+ "id": 20,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": true,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values\nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - stacked percent",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 57
+ },
+ "id": 14,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), CONCAT(measurement, ' - value one') as metric, valueOne FROM metric_values \nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), CONCAT(measurement, ' - value two') as metric, valueTwo FROM metric_values \nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - series mode",
+ "tooltip": {
+ "shared": false,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "series",
+ "name": null,
+ "show": true,
+ "values": [
+ "total"
+ ]
+ },
+ "yaxes": [
+ {
+ "decimals": null,
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MSSQL_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 57
+ },
+ "id": 15,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values\nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - series mode",
+ "tooltip": {
+ "shared": false,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "series",
+ "name": null,
+ "show": true,
+ "values": [
+ "total"
+ ]
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 65
+ },
+ "id": 25,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), CONCAT(measurement, ' - value one') as metric, valueOne FROM metric_values \nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), CONCAT(measurement, ' - value two') as metric, valueTwo FROM metric_values \nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - histogram",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 50,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "current"
+ ]
+ },
+ "yaxes": [
+ {
+ "decimals": null,
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 65
+ },
+ "id": 22,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values\nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - histogram",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 100,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "total"
+ ]
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 73
+ },
+ "id": 21,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), CONCAT(measurement, ' - value one') as metric, valueOne FROM metric_values \nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), CONCAT(measurement, ' - value two') as metric, valueTwo FROM metric_values \nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - histogram stacked",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 20,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "current"
+ ]
+ },
+ "yaxes": [
+ {
+ "decimals": null,
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 73
+ },
+ "id": 26,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values\nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - histogram stacked",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 20,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "total"
+ ]
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 81
+ },
+ "id": 23,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": true,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), CONCAT(measurement, ' - value one') as metric, valueOne FROM metric_values \nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), CONCAT(measurement, ' - value two') as metric, valueTwo FROM metric_values \nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - histogram stacked percent",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 20,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "current"
+ ]
+ },
+ "yaxes": [
+ {
+ "decimals": null,
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_MYSQL_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 81
+ },
+ "id": 24,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": true,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), valueOne, valueTwo FROM metric_values\nWHERE $__timeFilter(time) AND measurement IN($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - histogram stacked percent",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 20,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "total"
+ ]
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ }
+ ],
+ "refresh": false,
+ "schemaVersion": 16,
+ "style": "dark",
+ "tags": [],
+ "templating": {
+ "list": [
+ {
+ "allValue": "",
+ "current": {},
+ "datasource": "${DS_MYSQL_TEST}",
+ "hide": 0,
+ "includeAll": true,
+ "label": "Metric",
+ "multi": true,
+ "name": "metric",
+ "options": [],
+ "query": "SELECT DISTINCT measurement FROM metric_values",
+ "refresh": 1,
+ "regex": "",
+ "sort": 0,
+ "tagValuesQuery": "",
+ "tags": [],
+ "tagsQuery": "",
+ "type": "query",
+ "useTags": false
+ },
+ {
+ "auto": false,
+ "auto_count": 30,
+ "auto_min": "10s",
+ "current": {
+ "text": "10m",
+ "value": "10m"
+ },
+ "hide": 0,
+ "label": "Interval",
+ "name": "summarize",
+ "options": [
+ {
+ "selected": false,
+ "text": "1s",
+ "value": "1s"
+ },
+ {
+ "selected": false,
+ "text": "10s",
+ "value": "10s"
+ },
+ {
+ "selected": false,
+ "text": "30s",
+ "value": "30s"
+ },
+ {
+ "selected": false,
+ "text": "1m",
+ "value": "1m"
+ },
+ {
+ "selected": false,
+ "text": "5m",
+ "value": "5m"
+ },
+ {
+ "selected": true,
+ "text": "10m",
+ "value": "10m"
+ }
+ ],
+ "query": "1s,10s,30s,1m,5m,10m",
+ "refresh": 2,
+ "type": "interval"
+ }
+ ]
+ },
+ "time": {
+ "from": "2018-03-15T11:30:00.000Z",
+ "to": "2018-03-15T12:55:01.000Z"
+ },
+ "timepicker": {
+ "refresh_intervals": [
+ "5s",
+ "10s",
+ "30s",
+ "1m",
+ "5m",
+ "15m",
+ "30m",
+ "1h",
+ "2h",
+ "1d"
+ ],
+ "time_options": [
+ "5m",
+ "15m",
+ "1h",
+ "6h",
+ "12h",
+ "24h",
+ "2d",
+ "7d",
+ "30d"
+ ]
+ },
+ "timezone": "",
+ "title": "MySQL Data Source Test",
+ "uid": "Hmf8FDkmz",
+ "version": 9
+}
\ No newline at end of file
diff --git a/docker/blocks/postgres_tests/dashboard.json b/docker/blocks/postgres_tests/dashboard.json
new file mode 100644
index 00000000000..eea95863716
--- /dev/null
+++ b/docker/blocks/postgres_tests/dashboard.json
@@ -0,0 +1,2324 @@
+{
+ "__inputs": [
+ {
+ "name": "DS_POSTGRES_TEST",
+ "label": "Postgres TEST",
+ "description": "",
+ "type": "datasource",
+ "pluginId": "postgres",
+ "pluginName": "PostgreSQL"
+ }
+ ],
+ "__requires": [
+ {
+ "type": "grafana",
+ "id": "grafana",
+ "name": "Grafana",
+ "version": "5.0.0"
+ },
+ {
+ "type": "panel",
+ "id": "graph",
+ "name": "Graph",
+ "version": "5.0.0"
+ },
+ {
+ "type": "datasource",
+ "id": "postgres",
+ "name": "PostgreSQL",
+ "version": "5.0.0"
+ },
+ {
+ "type": "panel",
+ "id": "table",
+ "name": "Table",
+ "version": "5.0.0"
+ }
+ ],
+ "annotations": {
+ "list": [
+ {
+ "builtIn": 1,
+ "datasource": "-- Grafana --",
+ "enable": true,
+ "hide": true,
+ "iconColor": "rgba(0, 211, 255, 1)",
+ "name": "Annotations & Alerts",
+ "type": "dashboard"
+ },
+ {
+ "datasource": "${DS_POSTGRES_TEST}",
+ "enable": false,
+ "hide": false,
+ "iconColor": "#6ed0e0",
+ "limit": 100,
+ "name": "Deploys",
+ "rawQuery": "SELECT \"time_sec\" as time, description as text, tags FROM event WHERE $__unixEpochFilter(time_sec) AND tags='deploy' ORDER BY 1 ASC",
+ "showIn": 0,
+ "tags": [],
+ "type": "tags"
+ },
+ {
+ "datasource": "${DS_POSTGRES_TEST}",
+ "enable": false,
+ "hide": false,
+ "iconColor": "rgba(255, 96, 96, 1)",
+ "limit": 100,
+ "name": "Tickets",
+ "rawQuery": "SELECT \"time_sec\" as time, description as text, tags FROM event WHERE $__unixEpochFilter(time_sec) AND tags='ticket' ORDER BY 1 ASC",
+ "showIn": 0,
+ "tags": [],
+ "type": "tags"
+ },
+ {
+ "datasource": "${DS_POSTGRES_TEST}",
+ "enable": false,
+ "hide": false,
+ "iconColor": "#7eb26d",
+ "limit": 100,
+ "name": "Metric Values timeEpoch macro",
+ "rawQuery": "SELECT \n $__timeEpoch(time), \n measurement as text, \n '' as tags\nFROM\n metric_values \nWHERE\n $__timeFilter(time)\nORDER BY 1",
+ "showIn": 0,
+ "tags": [],
+ "type": "tags"
+ },
+ {
+ "datasource": "${DS_POSTGRES_TEST}",
+ "enable": false,
+ "hide": false,
+ "iconColor": "#1f78c1",
+ "limit": 100,
+ "name": "Metric Values native time",
+ "rawQuery": "SELECT \n time, \n measurement as text, \n '' as tags\nFROM\n metric_values \nWHERE\n $__timeFilter(time)\nORDER BY 1",
+ "showIn": 0,
+ "tags": [],
+ "type": "tags"
+ }
+ ]
+ },
+ "editable": true,
+ "gnetId": null,
+ "graphTooltip": 0,
+ "id": null,
+ "iteration": 1521725946837,
+ "links": [],
+ "panels": [
+ {
+ "columns": [],
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 4,
+ "w": 24,
+ "x": 0,
+ "y": 0
+ },
+ "id": 2,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 1,
+ "desc": false
+ },
+ "styles": [
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "string",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT * FROM postgres_types",
+ "refId": "A"
+ }
+ ],
+ "title": "Data types",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "columns": [],
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 0,
+ "y": 4
+ },
+ "id": 32,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT cast(null as bigint) as time",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "cast(null as bigint) as time",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "columns": [],
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 6,
+ "y": 4
+ },
+ "id": 33,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT cast(null as timestamp) as time",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "cast(null as datetime) as time",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "columns": [],
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 12,
+ "y": 4
+ },
+ "id": 34,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT localtimestamp as time",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "localtimestamp as time",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "columns": [],
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fontSize": "100%",
+ "gridPos": {
+ "h": 3,
+ "w": 6,
+ "x": 18,
+ "y": 4
+ },
+ "id": 35,
+ "links": [],
+ "pageSize": null,
+ "scroll": true,
+ "showHeader": true,
+ "sort": {
+ "col": 0,
+ "desc": true
+ },
+ "styles": [
+ {
+ "alias": "Time",
+ "dateFormat": "YYYY-MM-DD HH:mm:ss",
+ "pattern": "time",
+ "type": "date"
+ },
+ {
+ "alias": "",
+ "colorMode": null,
+ "colors": [
+ "rgba(245, 54, 54, 0.9)",
+ "rgba(237, 129, 40, 0.89)",
+ "rgba(50, 172, 45, 0.97)"
+ ],
+ "decimals": 2,
+ "pattern": "/.*/",
+ "thresholds": [],
+ "type": "number",
+ "unit": "short"
+ }
+ ],
+ "targets": [
+ {
+ "alias": "",
+ "format": "table",
+ "rawSql": "SELECT NOW() as time",
+ "refId": "A",
+ "target": ""
+ }
+ ],
+ "title": "NOW() as time",
+ "transform": "table",
+ "type": "table"
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 0,
+ "y": 7
+ },
+ "id": 7,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": true,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '5m'), avg(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "timeGroup macro 5m without fill",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 8,
+ "y": 7
+ },
+ "id": 9,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null as zero",
+ "percentage": false,
+ "pointradius": 3,
+ "points": true,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '5m', NULL), avg(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "timeGroup macro 5m with fill(NULL) and null as zero",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 16,
+ "y": 7
+ },
+ "id": 10,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": true,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '5m', 10.0), avg(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "timeGroup macro 5m with fill(10.0)",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 0,
+ "y": 16
+ },
+ "id": 16,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": false,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '$summarize'), avg(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Metrics - timeGroup macro $summarize without fill",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 8,
+ "y": 16
+ },
+ "id": 12,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": false,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null as zero",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '$summarize', NULL), sum(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Metrics - timeGroup macro $summarize with fill(NULL)",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 9,
+ "w": 8,
+ "x": 16,
+ "y": 16
+ },
+ "id": 13,
+ "legend": {
+ "avg": false,
+ "current": false,
+ "max": false,
+ "min": false,
+ "show": true,
+ "total": false,
+ "values": false
+ },
+ "lines": false,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": true,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeGroup(time, '$summarize', 100.0), sum(value) as value FROM metric WHERE $__timeFilter(time) GROUP BY 1 ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Metrics - timeGroup macro $summarize with fill(100.0)",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 25
+ },
+ "id": 27,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "hideEmpty": false,
+ "hideZero": false,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT \n $__timeGroup(time, '$summarize'), \n measurement || ' - value one' as metric, \n avg(\"valueOne\") as \"valueOne\"\nFROM\n metric_values \nWHERE\n $__timeFilter(time) AND\n measurement in($metric)\nGROUP BY 1, 2\nORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT \n $__timeGroup(time, '$summarize'), \n measurement || ' - value two' as metric, \n avg(\"valueTwo\") as \"valueTwo\"\nFROM\n metric_values \nWHERE\n $__timeFilter(time) AND\n measurement in($metric)\nGROUP BY 1, 2\nORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column using timeGroup macro ($summarize)",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 25
+ },
+ "id": 5,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT \n $__timeGroup(time, '$summarize'), \n avg(\"valueOne\") as \"valueOne\", \n avg(\"valueTwo\") as \"valueTwo\" \nFROM\n metric_values \nWHERE\n $__timeFilter(time) AND\n measurement in($metric)\nGROUP BY 1\nORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column using timeGroup macro ($summarize)",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 33
+ },
+ "id": 4,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "hideEmpty": false,
+ "hideZero": false,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value one' as metric, \"valueOne\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value two' as metric, \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 33
+ },
+ "id": 28,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), \"valueOne\", \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 41
+ },
+ "id": 19,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "hideEmpty": false,
+ "hideZero": false,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value one' as metric, \"valueOne\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value two' as metric, \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - stacked",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 41
+ },
+ "id": 18,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), \"valueOne\", \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - stacked",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 49
+ },
+ "id": 17,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "hideEmpty": false,
+ "hideZero": false,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": true,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value one' as metric, \"valueOne\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value two' as metric, \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - stacked percent",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": false,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 2,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 49
+ },
+ "id": 20,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": true,
+ "linewidth": 2,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": true,
+ "pointradius": 3,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), \"valueOne\", \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - stacked percent",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "time",
+ "name": null,
+ "show": true,
+ "values": []
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 57
+ },
+ "id": 14,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value one' as metric, \"valueOne\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value two' as metric, \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - series mode",
+ "tooltip": {
+ "shared": false,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "series",
+ "name": null,
+ "show": true,
+ "values": [
+ "total"
+ ]
+ },
+ "yaxes": [
+ {
+ "decimals": null,
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 57
+ },
+ "id": 15,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": true,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), \"valueOne\", \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - series mode",
+ "tooltip": {
+ "shared": false,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": null,
+ "mode": "series",
+ "name": null,
+ "show": true,
+ "values": [
+ "total"
+ ]
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 65
+ },
+ "id": 25,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value one' as metric, \"valueOne\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value two' as metric, \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - histogram",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 50,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "current"
+ ]
+ },
+ "yaxes": [
+ {
+ "decimals": null,
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 65
+ },
+ "id": 22,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": false,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), \"valueOne\", \"valueTwo\" FROM metric_values\nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - histogram",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 100,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "total"
+ ]
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 73
+ },
+ "id": 21,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value one' as metric, \"valueOne\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value two' as metric, \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - histogram stacked",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 20,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "current"
+ ]
+ },
+ "yaxes": [
+ {
+ "decimals": null,
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 73
+ },
+ "id": 26,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": false,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), \"valueOne\", \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - histogram stacked",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 20,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "total"
+ ]
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 0,
+ "y": 81
+ },
+ "id": 23,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": true,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value one' as metric, \"valueOne\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ },
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), measurement || ' - value two' as metric, \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "B"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series with metric column - histogram stacked percent",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 20,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "current"
+ ]
+ },
+ "yaxes": [
+ {
+ "decimals": null,
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ },
+ {
+ "aliasColors": {},
+ "bars": true,
+ "dashLength": 10,
+ "dashes": false,
+ "datasource": "${DS_POSTGRES_TEST}",
+ "fill": 1,
+ "gridPos": {
+ "h": 8,
+ "w": 12,
+ "x": 12,
+ "y": 81
+ },
+ "id": 24,
+ "legend": {
+ "alignAsTable": true,
+ "avg": true,
+ "current": true,
+ "max": true,
+ "min": true,
+ "rightSide": true,
+ "show": false,
+ "total": true,
+ "values": true
+ },
+ "lines": false,
+ "linewidth": 1,
+ "links": [],
+ "nullPointMode": "null",
+ "percentage": true,
+ "pointradius": 5,
+ "points": false,
+ "renderer": "flot",
+ "seriesOverrides": [],
+ "spaceLength": 10,
+ "stack": true,
+ "steppedLine": false,
+ "targets": [
+ {
+ "alias": "",
+ "format": "time_series",
+ "rawSql": "SELECT $__timeEpoch(time), \"valueOne\", \"valueTwo\" FROM metric_values \nWHERE $__timeFilter(time) AND measurement in($metric) ORDER BY 1",
+ "refId": "A"
+ }
+ ],
+ "thresholds": [],
+ "timeFrom": null,
+ "timeShift": null,
+ "title": "Multiple series without metric column - histogram stacked percent",
+ "tooltip": {
+ "shared": true,
+ "sort": 0,
+ "value_type": "individual"
+ },
+ "type": "graph",
+ "xaxis": {
+ "buckets": 20,
+ "mode": "histogram",
+ "name": null,
+ "show": true,
+ "values": [
+ "total"
+ ]
+ },
+ "yaxes": [
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": "0",
+ "show": true
+ },
+ {
+ "format": "short",
+ "label": null,
+ "logBase": 1,
+ "max": null,
+ "min": null,
+ "show": true
+ }
+ ]
+ }
+ ],
+ "refresh": false,
+ "schemaVersion": 16,
+ "style": "dark",
+ "tags": [],
+ "templating": {
+ "list": [
+ {
+ "allValue": null,
+ "current": {},
+ "datasource": "${DS_POSTGRES_TEST}",
+ "hide": 0,
+ "includeAll": true,
+ "label": "Metric",
+ "multi": true,
+ "name": "metric",
+ "options": [],
+ "query": "SELECT DISTINCT measurement FROM metric_values",
+ "refresh": 1,
+ "regex": "",
+ "sort": 1,
+ "tagValuesQuery": "",
+ "tags": [],
+ "tagsQuery": "",
+ "type": "query",
+ "useTags": false
+ },
+ {
+ "auto": false,
+ "auto_count": 30,
+ "auto_min": "10s",
+ "current": {
+ "text": "10m",
+ "value": "10m"
+ },
+ "hide": 0,
+ "label": "Interval",
+ "name": "summarize",
+ "options": [
+ {
+ "selected": false,
+ "text": "1s",
+ "value": "1s"
+ },
+ {
+ "selected": false,
+ "text": "10s",
+ "value": "10s"
+ },
+ {
+ "selected": false,
+ "text": "30s",
+ "value": "30s"
+ },
+ {
+ "selected": false,
+ "text": "1m",
+ "value": "1m"
+ },
+ {
+ "selected": false,
+ "text": "5m",
+ "value": "5m"
+ },
+ {
+ "selected": true,
+ "text": "10m",
+ "value": "10m"
+ }
+ ],
+ "query": "1s,10s,30s,1m,5m,10m",
+ "refresh": 2,
+ "type": "interval"
+ }
+ ]
+ },
+ "time": {
+ "from": "2018-03-15T12:30:00.000Z",
+ "to": "2018-03-15T13:55:01.000Z"
+ },
+ "timepicker": {
+ "refresh_intervals": [
+ "5s",
+ "10s",
+ "30s",
+ "1m",
+ "5m",
+ "15m",
+ "30m",
+ "1h",
+ "2h",
+ "1d"
+ ],
+ "time_options": [
+ "5m",
+ "15m",
+ "1h",
+ "6h",
+ "12h",
+ "24h",
+ "2d",
+ "7d",
+ "30d"
+ ]
+ },
+ "timezone": "",
+ "title": "Postgres Data Source Test",
+ "uid": "vHQdlVziz",
+ "version": 14
+}
\ No newline at end of file
diff --git a/docs/sources/administration/provisioning.md b/docs/sources/administration/provisioning.md
index a8cb9fe3023..b43ea68bcd8 100644
--- a/docs/sources/administration/provisioning.md
+++ b/docs/sources/administration/provisioning.md
@@ -11,11 +11,13 @@ weight = 8
# Provisioning Grafana
-## Config file
+In previous versions of Grafana, you could only use the API for provisioning data sources and dashboards. But that required the service to be running before you started creating dashboards and you also needed to set up credentials for the HTTP API. In v5.0 we decided to improve this experience by adding a new active provisioning system that uses config files. This will make GitOps more natural as data sources and dashboards can be defined via files that can be version controlled. We hope to extend this system to later add support for users, orgs and alerts as well.
+
+## Config File
Checkout the [configuration](/installation/configuration) page for more information on what you can configure in `grafana.ini`
-### Config file locations
+### Config File Locations
- Default configuration from `$WORKING_DIR/conf/defaults.ini`
- Custom configuration from `$WORKING_DIR/conf/custom.ini`
@@ -26,7 +28,7 @@ Checkout the [configuration](/installation/configuration) page for more informat
> `/etc/grafana/grafana.ini`. This path is specified in the Grafana
> init.d script using `--config` file parameter.
-### Using environment variables
+### Using Environment Variables
All options in the configuration file (listed below) can be overridden
using environment variables using the syntax:
@@ -59,7 +61,7 @@ export GF_AUTH_GOOGLE_CLIENT_SECRET=newS3cretKey
-## Configuration management tools
+## Configuration Management Tools
Currently we do not provide any scripts/manifests for configuring Grafana. Rather than spending time learning and creating scripts/manifests for each tool, we think our time is better spent making Grafana easier to provision. Therefore, we heavily relay on the expertise of the community.
@@ -76,10 +78,12 @@ Saltstack | [https://github.com/salt-formulas/salt-formula-grafana](https://gith
It's possible to manage datasources in Grafana by adding one or more yaml config files in the [`provisioning/datasources`](/installation/configuration/#provisioning) directory. Each config file can contain a list of `datasources` that will be added or updated during start up. If the datasource already exists, Grafana will update it to match the configuration file. The config file can also contain a list of datasources that should be deleted. That list is called `delete_datasources`. Grafana will delete datasources listed in `delete_datasources` before inserting/updating those in the `datasource` list.
-### Running multiple Grafana instances.
+### Running Multiple Grafana Instances
+
If you are running multiple instances of Grafana you might run into problems if they have different versions of the `datasource.yaml` configuration file. The best way to solve this problem is to add a version number to each datasource in the configuration and increase it when you update the config. Grafana will only update datasources with the same or lower version number than specified in the config. That way, old configs cannot overwrite newer configs if they restart at the same time.
-### Example datasource config file
+### Example Datasource Config File
+
```yaml
# config file version
apiVersion: 1
@@ -133,20 +137,20 @@ datasources:
editable: false
```
-#### Extra info per datasource
+#### Custom Settings per Datasource
| Datasource | Misc |
| ---- | ---- |
-| Elasticserach | Elasticsearch uses the `database` property to configure the index for a datasource |
+| Elasticsearch | Elasticsearch uses the `database` property to configure the index for a datasource |
-#### Json data
+#### Json Data
Since not all datasources have the same configuration settings we only have the most common ones as fields. The rest should be stored as a json blob in the `json_data` field. Here are the most common settings that the core datasources use.
| Name | Type | Datasource | Description |
| ---- | ---- | ---- | ---- |
| tlsAuth | boolean | *All* | Enable TLS authentication using client cert configured in secure json data |
-| tlsAuthWithCACert | boolean | *All* | Enable TLS authtication using CA cert |
+| tlsAuthWithCACert | boolean | *All* | Enable TLS authentication using CA cert |
| tlsSkipVerify | boolean | *All* | Controls whether a client verifies the server's certificate chain and host name. |
| graphiteVersion | string | Graphite | Graphite version |
| timeInterval | string | Elastic, Influxdb & Prometheus | Lowest interval/step value that should be used for this data source |
@@ -161,8 +165,7 @@ Since not all datasources have the same configuration settings we only have the
| tsdbResolution | string | OpenTsdb | Resolution |
| sslmode | string | Postgre | SSLmode. 'disable', 'require', 'verify-ca' or 'verify-full' |
-
-#### Secure Json data
+#### Secure Json Data
`{"authType":"keys","defaultRegion":"us-west-2","timeField":"@timestamp"}`
@@ -200,7 +203,7 @@ providers:
When Grafana starts, it will update/insert all dashboards available in the configured path. Then later on poll that path and look for updated json files and insert those update/insert those into the database.
-### Reuseable dashboard urls
+### Reuseable Dashboard Urls
If the dashboard in the json file contains an [uid](/reference/dashboard/#json-fields), Grafana will force insert/update on that uid. This allows you to migrate dashboards betweens Grafana instances and provisioning Grafana from configuration without breaking the urls given since the new dashboard url uses the uid as identifer.
When Grafana starts, it will update/insert all dashboards available in the configured folders. If you modify the file, the dashboard will also be updated.
diff --git a/pkg/api/alerting.go b/pkg/api/alerting.go
index 803823f6a94..a9a3773ceb1 100644
--- a/pkg/api/alerting.go
+++ b/pkg/api/alerting.go
@@ -29,7 +29,7 @@ func GetAlertStatesForDashboard(c *m.ReqContext) Response {
dashboardID := c.QueryInt64("dashboardId")
if dashboardID == 0 {
- return ApiError(400, "Missing query parameter dashboardId", nil)
+ return Error(400, "Missing query parameter dashboardId", nil)
}
query := m.GetAlertStatesForDashboardQuery{
@@ -38,10 +38,10 @@ func GetAlertStatesForDashboard(c *m.ReqContext) Response {
}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Failed to fetch alert states", err)
+ return Error(500, "Failed to fetch alert states", err)
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
// GET /api/alerts
@@ -60,20 +60,20 @@ func GetAlerts(c *m.ReqContext) Response {
}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "List alerts failed", err)
+ return Error(500, "List alerts failed", err)
}
for _, alert := range query.Result {
alert.Url = m.GetDashboardUrl(alert.DashboardUid, alert.DashboardSlug)
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
// POST /api/alerts/test
func AlertTest(c *m.ReqContext, dto dtos.AlertTestCommand) Response {
if _, idErr := dto.Dashboard.Get("id").Int64(); idErr != nil {
- return ApiError(400, "The dashboard needs to be saved at least once before you can test an alert rule", nil)
+ return Error(400, "The dashboard needs to be saved at least once before you can test an alert rule", nil)
}
backendCmd := alerting.AlertTestCommand{
@@ -84,9 +84,9 @@ func AlertTest(c *m.ReqContext, dto dtos.AlertTestCommand) Response {
if err := bus.Dispatch(&backendCmd); err != nil {
if validationErr, ok := err.(alerting.ValidationError); ok {
- return ApiError(422, validationErr.Error(), nil)
+ return Error(422, validationErr.Error(), nil)
}
- return ApiError(500, "Failed to test rule", err)
+ return Error(500, "Failed to test rule", err)
}
res := backendCmd.Result
@@ -109,7 +109,7 @@ func AlertTest(c *m.ReqContext, dto dtos.AlertTestCommand) Response {
dtoRes.TimeMs = fmt.Sprintf("%1.3fms", res.GetDurationMs())
- return Json(200, dtoRes)
+ return JSON(200, dtoRes)
}
// GET /api/alerts/:id
@@ -118,21 +118,21 @@ func GetAlert(c *m.ReqContext) Response {
query := m.GetAlertByIdQuery{Id: id}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "List alerts failed", err)
+ return Error(500, "List alerts failed", err)
}
- return Json(200, &query.Result)
+ return JSON(200, &query.Result)
}
func GetAlertNotifiers(c *m.ReqContext) Response {
- return Json(200, alerting.GetNotifiers())
+ return JSON(200, alerting.GetNotifiers())
}
func GetAlertNotifications(c *m.ReqContext) Response {
query := &m.GetAllAlertNotificationsQuery{OrgId: c.OrgId}
if err := bus.Dispatch(query); err != nil {
- return ApiError(500, "Failed to get alert notifications", err)
+ return Error(500, "Failed to get alert notifications", err)
}
result := make([]*dtos.AlertNotification, 0)
@@ -148,7 +148,7 @@ func GetAlertNotifications(c *m.ReqContext) Response {
})
}
- return Json(200, result)
+ return JSON(200, result)
}
func GetAlertNotificationByID(c *m.ReqContext) Response {
@@ -158,30 +158,30 @@ func GetAlertNotificationByID(c *m.ReqContext) Response {
}
if err := bus.Dispatch(query); err != nil {
- return ApiError(500, "Failed to get alert notifications", err)
+ return Error(500, "Failed to get alert notifications", err)
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
func CreateAlertNotification(c *m.ReqContext, cmd m.CreateAlertNotificationCommand) Response {
cmd.OrgId = c.OrgId
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to create alert notification", err)
+ return Error(500, "Failed to create alert notification", err)
}
- return Json(200, cmd.Result)
+ return JSON(200, cmd.Result)
}
func UpdateAlertNotification(c *m.ReqContext, cmd m.UpdateAlertNotificationCommand) Response {
cmd.OrgId = c.OrgId
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to update alert notification", err)
+ return Error(500, "Failed to update alert notification", err)
}
- return Json(200, cmd.Result)
+ return JSON(200, cmd.Result)
}
func DeleteAlertNotification(c *m.ReqContext) Response {
@@ -191,10 +191,10 @@ func DeleteAlertNotification(c *m.ReqContext) Response {
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to delete alert notification", err)
+ return Error(500, "Failed to delete alert notification", err)
}
- return ApiSuccess("Notification deleted")
+ return Success("Notification deleted")
}
//POST /api/alert-notifications/test
@@ -207,41 +207,41 @@ func NotificationTest(c *m.ReqContext, dto dtos.NotificationTestCommand) Respons
if err := bus.Dispatch(cmd); err != nil {
if err == m.ErrSmtpNotEnabled {
- return ApiError(412, err.Error(), err)
+ return Error(412, err.Error(), err)
}
- return ApiError(500, "Failed to send alert notifications", err)
+ return Error(500, "Failed to send alert notifications", err)
}
- return ApiSuccess("Test notification sent")
+ return Success("Test notification sent")
}
//POST /api/alerts/:alertId/pause
func PauseAlert(c *m.ReqContext, dto dtos.PauseAlertCommand) Response {
- alertId := c.ParamsInt64("alertId")
+ alertID := c.ParamsInt64("alertId")
- query := m.GetAlertByIdQuery{Id: alertId}
+ query := m.GetAlertByIdQuery{Id: alertID}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Get Alert failed", err)
+ return Error(500, "Get Alert failed", err)
}
guardian := guardian.New(query.Result.DashboardId, c.OrgId, c.SignedInUser)
if canEdit, err := guardian.CanEdit(); err != nil || !canEdit {
if err != nil {
- return ApiError(500, "Error while checking permissions for Alert", err)
+ return Error(500, "Error while checking permissions for Alert", err)
}
- return ApiError(403, "Access denied to this dashboard and alert", nil)
+ return Error(403, "Access denied to this dashboard and alert", nil)
}
cmd := m.PauseAlertCommand{
OrgId: c.OrgId,
- AlertIds: []int64{alertId},
+ AlertIds: []int64{alertID},
Paused: dto.Paused,
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "", err)
+ return Error(500, "", err)
}
var response m.AlertStateType = m.AlertStatePending
@@ -252,12 +252,12 @@ func PauseAlert(c *m.ReqContext, dto dtos.PauseAlertCommand) Response {
}
result := map[string]interface{}{
- "alertId": alertId,
+ "alertId": alertID,
"state": response,
"message": "Alert " + pausedState,
}
- return Json(200, result)
+ return JSON(200, result)
}
//POST /api/admin/pause-all-alerts
@@ -267,7 +267,7 @@ func PauseAllAlerts(c *m.ReqContext, dto dtos.PauseAllAlertsCommand) Response {
}
if err := bus.Dispatch(&updateCmd); err != nil {
- return ApiError(500, "Failed to pause alerts", err)
+ return Error(500, "Failed to pause alerts", err)
}
var response m.AlertStateType = m.AlertStatePending
@@ -283,5 +283,5 @@ func PauseAllAlerts(c *m.ReqContext, dto dtos.PauseAllAlertsCommand) Response {
"alertsAffected": updateCmd.ResultCount,
}
- return Json(200, result)
+ return JSON(200, result)
}
diff --git a/pkg/api/annotations.go b/pkg/api/annotations.go
index b4e328793cc..886913d6324 100644
--- a/pkg/api/annotations.go
+++ b/pkg/api/annotations.go
@@ -30,7 +30,7 @@ func GetAnnotations(c *m.ReqContext) Response {
items, err := repo.Find(query)
if err != nil {
- return ApiError(500, "Failed to get annotations", err)
+ return Error(500, "Failed to get annotations", err)
}
for _, item := range items {
@@ -40,7 +40,7 @@ func GetAnnotations(c *m.ReqContext) Response {
item.Time = item.Time * 1000
}
- return Json(200, items)
+ return JSON(200, items)
}
type CreateAnnotationError struct {
@@ -60,7 +60,7 @@ func PostAnnotation(c *m.ReqContext, cmd dtos.PostAnnotationsCmd) Response {
if cmd.Text == "" {
err := &CreateAnnotationError{"text field should not be empty"}
- return ApiError(500, "Failed to save annotation", err)
+ return Error(500, "Failed to save annotation", err)
}
item := annotations.Item{
@@ -79,7 +79,7 @@ func PostAnnotation(c *m.ReqContext, cmd dtos.PostAnnotationsCmd) Response {
}
if err := repo.Save(&item); err != nil {
- return ApiError(500, "Failed to save annotation", err)
+ return Error(500, "Failed to save annotation", err)
}
startID := item.Id
@@ -93,24 +93,24 @@ func PostAnnotation(c *m.ReqContext, cmd dtos.PostAnnotationsCmd) Response {
}
if err := repo.Update(&item); err != nil {
- return ApiError(500, "Failed set regionId on annotation", err)
+ return Error(500, "Failed set regionId on annotation", err)
}
item.Id = 0
item.Epoch = cmd.TimeEnd / 1000
if err := repo.Save(&item); err != nil {
- return ApiError(500, "Failed save annotation for region end time", err)
+ return Error(500, "Failed save annotation for region end time", err)
}
- return Json(200, util.DynMap{
+ return JSON(200, util.DynMap{
"message": "Annotation added",
"id": startID,
"endId": item.Id,
})
}
- return Json(200, util.DynMap{
+ return JSON(200, util.DynMap{
"message": "Annotation added",
"id": startID,
})
@@ -129,7 +129,7 @@ func PostGraphiteAnnotation(c *m.ReqContext, cmd dtos.PostGraphiteAnnotationsCmd
if cmd.What == "" {
err := &CreateAnnotationError{"what field should not be empty"}
- return ApiError(500, "Failed to save Graphite annotation", err)
+ return Error(500, "Failed to save Graphite annotation", err)
}
if cmd.When == 0 {
@@ -152,12 +152,12 @@ func PostGraphiteAnnotation(c *m.ReqContext, cmd dtos.PostGraphiteAnnotationsCmd
tagsArray = append(tagsArray, tagStr)
} else {
err := &CreateAnnotationError{"tag should be a string"}
- return ApiError(500, "Failed to save Graphite annotation", err)
+ return Error(500, "Failed to save Graphite annotation", err)
}
}
default:
err := &CreateAnnotationError{"unsupported tags format"}
- return ApiError(500, "Failed to save Graphite annotation", err)
+ return Error(500, "Failed to save Graphite annotation", err)
}
item := annotations.Item{
@@ -169,10 +169,10 @@ func PostGraphiteAnnotation(c *m.ReqContext, cmd dtos.PostGraphiteAnnotationsCmd
}
if err := repo.Save(&item); err != nil {
- return ApiError(500, "Failed to save Graphite annotation", err)
+ return Error(500, "Failed to save Graphite annotation", err)
}
- return Json(200, util.DynMap{
+ return JSON(200, util.DynMap{
"message": "Graphite annotation added",
"id": item.Id,
})
@@ -197,7 +197,7 @@ func UpdateAnnotation(c *m.ReqContext, cmd dtos.UpdateAnnotationsCmd) Response {
}
if err := repo.Update(&item); err != nil {
- return ApiError(500, "Failed to update annotation", err)
+ return Error(500, "Failed to update annotation", err)
}
if cmd.IsRegion {
@@ -210,11 +210,11 @@ func UpdateAnnotation(c *m.ReqContext, cmd dtos.UpdateAnnotationsCmd) Response {
itemRight.Id = 0
if err := repo.Update(&itemRight); err != nil {
- return ApiError(500, "Failed to update annotation for region end time", err)
+ return Error(500, "Failed to update annotation for region end time", err)
}
}
- return ApiSuccess("Annotation updated")
+ return Success("Annotation updated")
}
func DeleteAnnotations(c *m.ReqContext, cmd dtos.DeleteAnnotationsCmd) Response {
@@ -227,29 +227,29 @@ func DeleteAnnotations(c *m.ReqContext, cmd dtos.DeleteAnnotationsCmd) Response
})
if err != nil {
- return ApiError(500, "Failed to delete annotations", err)
+ return Error(500, "Failed to delete annotations", err)
}
- return ApiSuccess("Annotations deleted")
+ return Success("Annotations deleted")
}
-func DeleteAnnotationById(c *m.ReqContext) Response {
+func DeleteAnnotationByID(c *m.ReqContext) Response {
repo := annotations.GetRepository()
- annotationId := c.ParamsInt64(":annotationId")
+ annotationID := c.ParamsInt64(":annotationId")
- if resp := canSave(c, repo, annotationId); resp != nil {
+ if resp := canSave(c, repo, annotationID); resp != nil {
return resp
}
err := repo.Delete(&annotations.DeleteParams{
- Id: annotationId,
+ Id: annotationID,
})
if err != nil {
- return ApiError(500, "Failed to delete annotation", err)
+ return Error(500, "Failed to delete annotation", err)
}
- return ApiSuccess("Annotation deleted")
+ return Success("Annotation deleted")
}
func DeleteAnnotationRegion(c *m.ReqContext) Response {
@@ -265,10 +265,10 @@ func DeleteAnnotationRegion(c *m.ReqContext) Response {
})
if err != nil {
- return ApiError(500, "Failed to delete annotation region", err)
+ return Error(500, "Failed to delete annotation region", err)
}
- return ApiSuccess("Annotation region deleted")
+ return Success("Annotation region deleted")
}
func canSaveByDashboardID(c *m.ReqContext, dashboardID int64) (bool, error) {
@@ -286,11 +286,11 @@ func canSaveByDashboardID(c *m.ReqContext, dashboardID int64) (bool, error) {
return true, nil
}
-func canSave(c *m.ReqContext, repo annotations.Repository, annotationId int64) Response {
- items, err := repo.Find(&annotations.ItemQuery{AnnotationId: annotationId, OrgId: c.OrgId})
+func canSave(c *m.ReqContext, repo annotations.Repository, annotationID int64) Response {
+ items, err := repo.Find(&annotations.ItemQuery{AnnotationId: annotationID, OrgId: c.OrgId})
if err != nil || len(items) == 0 {
- return ApiError(500, "Could not find annotation to update", err)
+ return Error(500, "Could not find annotation to update", err)
}
dashboardID := items[0].DashboardId
@@ -306,7 +306,7 @@ func canSaveByRegionID(c *m.ReqContext, repo annotations.Repository, regionID in
items, err := repo.Find(&annotations.ItemQuery{RegionId: regionID, OrgId: c.OrgId})
if err != nil || len(items) == 0 {
- return ApiError(500, "Could not find annotation to update", err)
+ return Error(500, "Could not find annotation to update", err)
}
dashboardID := items[0].DashboardId
diff --git a/pkg/api/annotations_test.go b/pkg/api/annotations_test.go
index 7c298550673..9fe96245b9b 100644
--- a/pkg/api/annotations_test.go
+++ b/pkg/api/annotations_test.go
@@ -41,7 +41,7 @@ func TestAnnotationsApiEndpoint(t *testing.T) {
})
loggedInUserScenarioWithRole("When calling DELETE on", "DELETE", "/api/annotations/1", "/api/annotations/:annotationId", role, func(sc *scenarioContext) {
- sc.handlerFunc = DeleteAnnotationById
+ sc.handlerFunc = DeleteAnnotationByID
sc.fakeReqWithParams("DELETE", sc.url, map[string]string{}).exec()
So(sc.resp.Code, ShouldEqual, 403)
})
@@ -68,7 +68,7 @@ func TestAnnotationsApiEndpoint(t *testing.T) {
})
loggedInUserScenarioWithRole("When calling DELETE on", "DELETE", "/api/annotations/1", "/api/annotations/:annotationId", role, func(sc *scenarioContext) {
- sc.handlerFunc = DeleteAnnotationById
+ sc.handlerFunc = DeleteAnnotationByID
sc.fakeReqWithParams("DELETE", sc.url, map[string]string{}).exec()
So(sc.resp.Code, ShouldEqual, 200)
})
@@ -132,7 +132,7 @@ func TestAnnotationsApiEndpoint(t *testing.T) {
})
loggedInUserScenarioWithRole("When calling DELETE on", "DELETE", "/api/annotations/1", "/api/annotations/:annotationId", role, func(sc *scenarioContext) {
- sc.handlerFunc = DeleteAnnotationById
+ sc.handlerFunc = DeleteAnnotationByID
sc.fakeReqWithParams("DELETE", sc.url, map[string]string{}).exec()
So(sc.resp.Code, ShouldEqual, 403)
})
@@ -159,7 +159,7 @@ func TestAnnotationsApiEndpoint(t *testing.T) {
})
loggedInUserScenarioWithRole("When calling DELETE on", "DELETE", "/api/annotations/1", "/api/annotations/:annotationId", role, func(sc *scenarioContext) {
- sc.handlerFunc = DeleteAnnotationById
+ sc.handlerFunc = DeleteAnnotationByID
sc.fakeReqWithParams("DELETE", sc.url, map[string]string{}).exec()
So(sc.resp.Code, ShouldEqual, 200)
})
diff --git a/pkg/api/api.go b/pkg/api/api.go
index 22d5d773d2a..3c7b81e472d 100644
--- a/pkg/api/api.go
+++ b/pkg/api/api.go
@@ -9,14 +9,14 @@ import (
)
// Register adds http routes
-func (hs *HttpServer) registerRoutes() {
+func (hs *HTTPServer) registerRoutes() {
macaronR := hs.macaron
reqSignedIn := middleware.Auth(&middleware.AuthOptions{ReqSignedIn: true})
reqGrafanaAdmin := middleware.Auth(&middleware.AuthOptions{ReqSignedIn: true, ReqGrafanaAdmin: true})
reqEditorRole := middleware.RoleAuth(m.ROLE_EDITOR, m.ROLE_ADMIN)
reqOrgAdmin := middleware.RoleAuth(m.ROLE_ADMIN)
- redirectFromLegacyDashboardUrl := middleware.RedirectFromLegacyDashboardURL()
- redirectFromLegacyDashboardSoloUrl := middleware.RedirectFromLegacyDashboardSoloUrl()
+ redirectFromLegacyDashboardURL := middleware.RedirectFromLegacyDashboardURL()
+ redirectFromLegacyDashboardSoloURL := middleware.RedirectFromLegacyDashboardSoloURL()
quota := middleware.Quota
bind := binding.Bind
@@ -67,11 +67,11 @@ func (hs *HttpServer) registerRoutes() {
r.Get("/d/:uid/:slug", reqSignedIn, Index)
r.Get("/d/:uid", reqSignedIn, Index)
- r.Get("/dashboard/db/:slug", reqSignedIn, redirectFromLegacyDashboardUrl, Index)
+ r.Get("/dashboard/db/:slug", reqSignedIn, redirectFromLegacyDashboardURL, Index)
r.Get("/dashboard/script/*", reqSignedIn, Index)
r.Get("/dashboard-solo/snapshot/*", Index)
r.Get("/d-solo/:uid/:slug", reqSignedIn, Index)
- r.Get("/dashboard-solo/db/:slug", reqSignedIn, redirectFromLegacyDashboardSoloUrl, Index)
+ r.Get("/dashboard-solo/db/:slug", reqSignedIn, redirectFromLegacyDashboardSoloURL, Index)
r.Get("/dashboard-solo/script/*", reqSignedIn, Index)
r.Get("/import/dashboard", reqSignedIn, Index)
r.Get("/dashboards/", reqSignedIn, Index)
@@ -341,7 +341,7 @@ func (hs *HttpServer) registerRoutes() {
apiRoute.Group("/annotations", func(annotationsRoute RouteRegister) {
annotationsRoute.Post("/", bind(dtos.PostAnnotationsCmd{}), wrap(PostAnnotation))
- annotationsRoute.Delete("/:annotationId", wrap(DeleteAnnotationById))
+ annotationsRoute.Delete("/:annotationId", wrap(DeleteAnnotationByID))
annotationsRoute.Put("/:annotationId", bind(dtos.UpdateAnnotationsCmd{}), wrap(UpdateAnnotation))
annotationsRoute.Delete("/region/:regionId", wrap(DeleteAnnotationRegion))
annotationsRoute.Post("/graphite", reqEditorRole, bind(dtos.PostGraphiteAnnotationsCmd{}), wrap(PostGraphiteAnnotation))
diff --git a/pkg/api/apikey.go b/pkg/api/apikey.go
index 7195c7453f6..7fda738f1cd 100644
--- a/pkg/api/apikey.go
+++ b/pkg/api/apikey.go
@@ -11,7 +11,7 @@ func GetAPIKeys(c *m.ReqContext) Response {
query := m.GetApiKeysQuery{OrgId: c.OrgId}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Failed to list api keys", err)
+ return Error(500, "Failed to list api keys", err)
}
result := make([]*m.ApiKeyDTO, len(query.Result))
@@ -23,7 +23,7 @@ func GetAPIKeys(c *m.ReqContext) Response {
}
}
- return Json(200, result)
+ return JSON(200, result)
}
func DeleteAPIKey(c *m.ReqContext) Response {
@@ -33,15 +33,15 @@ func DeleteAPIKey(c *m.ReqContext) Response {
err := bus.Dispatch(cmd)
if err != nil {
- return ApiError(500, "Failed to delete API key", err)
+ return Error(500, "Failed to delete API key", err)
}
- return ApiSuccess("API key deleted")
+ return Success("API key deleted")
}
func AddAPIKey(c *m.ReqContext, cmd m.AddApiKeyCommand) Response {
if !cmd.Role.IsValid() {
- return ApiError(400, "Invalid role specified", nil)
+ return Error(400, "Invalid role specified", nil)
}
cmd.OrgId = c.OrgId
@@ -50,12 +50,12 @@ func AddAPIKey(c *m.ReqContext, cmd m.AddApiKeyCommand) Response {
cmd.Key = newKeyInfo.HashedKey
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to add API key", err)
+ return Error(500, "Failed to add API key", err)
}
result := &dtos.NewApiKeyResult{
Name: cmd.Result.Name,
Key: newKeyInfo.ClientSecret}
- return Json(200, result)
+ return JSON(200, result)
}
diff --git a/pkg/api/common.go b/pkg/api/common.go
index 370f78f8b1d..97f41ff7c72 100644
--- a/pkg/api/common.go
+++ b/pkg/api/common.go
@@ -11,10 +11,10 @@ import (
var (
NotFound = func() Response {
- return ApiError(404, "Not found", nil)
+ return Error(404, "Not found", nil)
}
ServerError = func(err error) Response {
- return ApiError(500, "Server error", err)
+ return Error(500, "Server error", err)
}
)
@@ -67,22 +67,25 @@ func (r *NormalResponse) Header(key, value string) *NormalResponse {
return r
}
-// functions to create responses
+// Empty create an empty response
func Empty(status int) *NormalResponse {
return Respond(status, nil)
}
-func Json(status int, body interface{}) *NormalResponse {
+// JSON create a JSON response
+func JSON(status int, body interface{}) *NormalResponse {
return Respond(status, body).Header("Content-Type", "application/json")
}
-func ApiSuccess(message string) *NormalResponse {
+// Success create a successful response
+func Success(message string) *NormalResponse {
resp := make(map[string]interface{})
resp["message"] = message
- return Json(200, resp)
+ return JSON(200, resp)
}
-func ApiError(status int, message string, err error) *NormalResponse {
+// Error create a erroneous response
+func Error(status int, message string, err error) *NormalResponse {
data := make(map[string]interface{})
switch status {
@@ -102,7 +105,7 @@ func ApiError(status int, message string, err error) *NormalResponse {
}
}
- resp := Json(status, data)
+ resp := JSON(status, data)
if err != nil {
resp.errMessage = message
@@ -112,6 +115,7 @@ func ApiError(status int, message string, err error) *NormalResponse {
return resp
}
+// Respond create a response
func Respond(status int, body interface{}) *NormalResponse {
var b []byte
var err error
@@ -122,7 +126,7 @@ func Respond(status int, body interface{}) *NormalResponse {
b = []byte(t)
default:
if b, err = json.Marshal(body); err != nil {
- return ApiError(500, "body json marshal", err)
+ return Error(500, "body json marshal", err)
}
}
return &NormalResponse{
diff --git a/pkg/api/dashboard.go b/pkg/api/dashboard.go
index 25c89238933..11a028cdd29 100644
--- a/pkg/api/dashboard.go
+++ b/pkg/api/dashboard.go
@@ -37,10 +37,10 @@ func isDashboardStarredByUser(c *m.ReqContext, dashID int64) (bool, error) {
func dashboardGuardianResponse(err error) Response {
if err != nil {
- return ApiError(500, "Error while checking dashboard permissions", err)
+ return Error(500, "Error while checking dashboard permissions", err)
}
- return ApiError(403, "Access denied to this dashboard", nil)
+ return Error(403, "Access denied to this dashboard", nil)
}
func GetDashboard(c *m.ReqContext) Response {
@@ -60,7 +60,7 @@ func GetDashboard(c *m.ReqContext) Response {
isStarred, err := isDashboardStarredByUser(c, dash.Id)
if err != nil {
- return ApiError(500, "Error while checking if dashboard was starred by user", err)
+ return Error(500, "Error while checking if dashboard was starred by user", err)
}
// Finding creator and last updater of the dashboard
@@ -96,7 +96,7 @@ func GetDashboard(c *m.ReqContext) Response {
if dash.FolderId > 0 {
query := m.GetDashboardQuery{Id: dash.FolderId, OrgId: c.OrgId}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Dashboard folder could not be read", err)
+ return Error(500, "Dashboard folder could not be read", err)
}
meta.FolderTitle = query.Result.Title
meta.FolderUrl = query.Result.GetUrl()
@@ -111,7 +111,7 @@ func GetDashboard(c *m.ReqContext) Response {
}
c.TimeRequest(metrics.M_Api_Dashboard_Get)
- return Json(200, dto)
+ return JSON(200, dto)
}
func getUserLogin(userID int64) string {
@@ -133,7 +133,7 @@ func getDashboardHelper(orgID int64, slug string, id int64, uid string) (*m.Dash
}
if err := bus.Dispatch(&query); err != nil {
- return nil, ApiError(404, "Dashboard not found", err)
+ return nil, Error(404, "Dashboard not found", err)
}
return query.Result, nil
@@ -143,11 +143,11 @@ func DeleteDashboard(c *m.ReqContext) Response {
query := m.GetDashboardsBySlugQuery{OrgId: c.OrgId, Slug: c.Params(":slug")}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Failed to retrieve dashboards by slug", err)
+ return Error(500, "Failed to retrieve dashboards by slug", err)
}
if len(query.Result) > 1 {
- return Json(412, util.DynMap{"status": "multiple-slugs-exists", "message": m.ErrDashboardsWithSameSlugExists.Error()})
+ return JSON(412, util.DynMap{"status": "multiple-slugs-exists", "message": m.ErrDashboardsWithSameSlugExists.Error()})
}
dash, rsp := getDashboardHelper(c.OrgId, c.Params(":slug"), 0, "")
@@ -162,10 +162,10 @@ func DeleteDashboard(c *m.ReqContext) Response {
cmd := m.DeleteDashboardCommand{OrgId: c.OrgId, Id: dash.Id}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to delete dashboard", err)
+ return Error(500, "Failed to delete dashboard", err)
}
- return Json(200, util.DynMap{
+ return JSON(200, util.DynMap{
"title": dash.Title,
"message": fmt.Sprintf("Dashboard %s deleted", dash.Title),
})
@@ -184,10 +184,10 @@ func DeleteDashboardByUID(c *m.ReqContext) Response {
cmd := m.DeleteDashboardCommand{OrgId: c.OrgId, Id: dash.Id}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to delete dashboard", err)
+ return Error(500, "Failed to delete dashboard", err)
}
- return Json(200, util.DynMap{
+ return JSON(200, util.DynMap{
"title": dash.Title,
"message": fmt.Sprintf("Dashboard %s deleted", dash.Title),
})
@@ -202,10 +202,10 @@ func PostDashboard(c *m.ReqContext, cmd m.SaveDashboardCommand) Response {
if dash.Id == 0 && dash.Uid == "" {
limitReached, err := quota.QuotaReached(c, "dashboard")
if err != nil {
- return ApiError(500, "failed to get quota", err)
+ return Error(500, "failed to get quota", err)
}
if limitReached {
- return ApiError(403, "Quota reached", nil)
+ return Error(403, "Quota reached", nil)
}
}
@@ -229,23 +229,23 @@ func PostDashboard(c *m.ReqContext, cmd m.SaveDashboardCommand) Response {
err == m.ErrFolderNotFound ||
err == m.ErrDashboardFolderCannotHaveParent ||
err == m.ErrDashboardFolderNameExists {
- return ApiError(400, err.Error(), nil)
+ return Error(400, err.Error(), nil)
}
if err == m.ErrDashboardUpdateAccessDenied {
- return ApiError(403, err.Error(), err)
+ return Error(403, err.Error(), err)
}
if err == m.ErrDashboardContainsInvalidAlertData {
- return ApiError(500, "Invalid alert data. Cannot save dashboard", err)
+ return Error(500, "Invalid alert data. Cannot save dashboard", err)
}
if err != nil {
if err == m.ErrDashboardWithSameNameInFolderExists {
- return Json(412, util.DynMap{"status": "name-exists", "message": err.Error()})
+ return JSON(412, util.DynMap{"status": "name-exists", "message": err.Error()})
}
if err == m.ErrDashboardVersionMismatch {
- return Json(412, util.DynMap{"status": "version-mismatch", "message": err.Error()})
+ return JSON(412, util.DynMap{"status": "version-mismatch", "message": err.Error()})
}
if pluginErr, ok := err.(m.UpdatePluginDashboardError); ok {
message := "The dashboard belongs to plugin " + pluginErr.PluginId + "."
@@ -253,20 +253,20 @@ func PostDashboard(c *m.ReqContext, cmd m.SaveDashboardCommand) Response {
if pluginDef, exist := plugins.Plugins[pluginErr.PluginId]; exist {
message = "The dashboard belongs to plugin " + pluginDef.Name + "."
}
- return Json(412, util.DynMap{"status": "plugin-dashboard", "message": message})
+ return JSON(412, util.DynMap{"status": "plugin-dashboard", "message": message})
}
if err == m.ErrDashboardNotFound {
- return Json(404, util.DynMap{"status": "not-found", "message": err.Error()})
+ return JSON(404, util.DynMap{"status": "not-found", "message": err.Error()})
}
- return ApiError(500, "Failed to save dashboard", err)
+ return Error(500, "Failed to save dashboard", err)
}
if err == m.ErrDashboardFailedToUpdateAlertData {
- return ApiError(500, "Invalid alert data. Cannot save dashboard", err)
+ return Error(500, "Invalid alert data. Cannot save dashboard", err)
}
c.TimeRequest(metrics.M_Api_Dashboard_Save)
- return Json(200, util.DynMap{
+ return JSON(200, util.DynMap{
"status": "success",
"slug": dashboard.Slug,
"version": dashboard.Version,
@@ -279,7 +279,7 @@ func PostDashboard(c *m.ReqContext, cmd m.SaveDashboardCommand) Response {
func GetHomeDashboard(c *m.ReqContext) Response {
prefsQuery := m.GetPreferencesWithDefaultsQuery{OrgId: c.OrgId, UserId: c.UserId}
if err := bus.Dispatch(&prefsQuery); err != nil {
- return ApiError(500, "Failed to get preferences", err)
+ return Error(500, "Failed to get preferences", err)
}
if prefsQuery.Result.HomeDashboardId != 0 {
@@ -288,7 +288,7 @@ func GetHomeDashboard(c *m.ReqContext) Response {
if err == nil {
url := m.GetDashboardUrl(slugQuery.Result.Uid, slugQuery.Result.Slug)
dashRedirect := dtos.DashboardRedirect{RedirectUri: url}
- return Json(200, &dashRedirect)
+ return JSON(200, &dashRedirect)
}
log.Warn("Failed to get slug from database, %s", err.Error())
}
@@ -296,7 +296,7 @@ func GetHomeDashboard(c *m.ReqContext) Response {
filePath := path.Join(setting.StaticRootPath, "dashboards/home.json")
file, err := os.Open(filePath)
if err != nil {
- return ApiError(500, "Failed to load home dashboard", err)
+ return Error(500, "Failed to load home dashboard", err)
}
dash := dtos.DashboardFullWithMeta{}
@@ -306,14 +306,14 @@ func GetHomeDashboard(c *m.ReqContext) Response {
jsonParser := json.NewDecoder(file)
if err := jsonParser.Decode(&dash.Dashboard); err != nil {
- return ApiError(500, "Failed to load home dashboard", err)
+ return Error(500, "Failed to load home dashboard", err)
}
if c.HasUserRole(m.ROLE_ADMIN) && !c.HasHelpFlag(m.HelpFlagGettingStartedPanelDismissed) {
addGettingStartedPanelToHomeDashboard(dash.Dashboard)
}
- return Json(200, &dash)
+ return JSON(200, &dash)
}
func addGettingStartedPanelToHomeDashboard(dash *simplejson.Json) {
@@ -351,7 +351,7 @@ func GetDashboardVersions(c *m.ReqContext) Response {
}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(404, fmt.Sprintf("No versions found for dashboardId %d", dashID), err)
+ return Error(404, fmt.Sprintf("No versions found for dashboardId %d", dashID), err)
}
for _, version := range query.Result {
@@ -370,7 +370,7 @@ func GetDashboardVersions(c *m.ReqContext) Response {
}
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
// GetDashboardVersion returns the dashboard version with the given ID.
@@ -389,7 +389,7 @@ func GetDashboardVersion(c *m.ReqContext) Response {
}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, fmt.Sprintf("Dashboard version %d not found for dashboardId %d", query.Version, dashID), err)
+ return Error(500, fmt.Sprintf("Dashboard version %d not found for dashboardId %d", query.Version, dashID), err)
}
creator := "Anonymous"
@@ -402,7 +402,7 @@ func GetDashboardVersion(c *m.ReqContext) Response {
CreatedBy: creator,
}
- return Json(200, dashVersionMeta)
+ return JSON(200, dashVersionMeta)
}
// POST /api/dashboards/calculate-diff performs diffs on two dashboards
@@ -438,9 +438,9 @@ func CalculateDashboardDiff(c *m.ReqContext, apiOptions dtos.CalculateDiffOption
result, err := dashdiffs.CalculateDiff(&options)
if err != nil {
if err == m.ErrDashboardVersionNotFound {
- return ApiError(404, "Dashboard version not found", err)
+ return Error(404, "Dashboard version not found", err)
}
- return ApiError(500, "Unable to compute diff", err)
+ return Error(500, "Unable to compute diff", err)
}
if options.DiffType == dashdiffs.DiffDelta {
@@ -464,7 +464,7 @@ func RestoreDashboardVersion(c *m.ReqContext, apiCmd dtos.RestoreDashboardVersio
versionQuery := m.GetDashboardVersionQuery{DashboardId: dash.Id, Version: apiCmd.Version, OrgId: c.OrgId}
if err := bus.Dispatch(&versionQuery); err != nil {
- return ApiError(404, "Dashboard version not found", nil)
+ return Error(404, "Dashboard version not found", nil)
}
version := versionQuery.Result
diff --git a/pkg/api/dashboard_permission.go b/pkg/api/dashboard_permission.go
index c852033829a..653815aea5c 100644
--- a/pkg/api/dashboard_permission.go
+++ b/pkg/api/dashboard_permission.go
@@ -25,7 +25,7 @@ func GetDashboardPermissionList(c *m.ReqContext) Response {
acl, err := g.GetAcl()
if err != nil {
- return ApiError(500, "Failed to get dashboard permissions", err)
+ return Error(500, "Failed to get dashboard permissions", err)
}
for _, perm := range acl {
@@ -34,7 +34,7 @@ func GetDashboardPermissionList(c *m.ReqContext) Response {
}
}
- return Json(200, acl)
+ return JSON(200, acl)
}
func UpdateDashboardPermissions(c *m.ReqContext, apiCmd dtos.UpdateDashboardAclCommand) Response {
@@ -70,21 +70,21 @@ func UpdateDashboardPermissions(c *m.ReqContext, apiCmd dtos.UpdateDashboardAclC
if err != nil {
if err == guardian.ErrGuardianPermissionExists ||
err == guardian.ErrGuardianOverride {
- return ApiError(400, err.Error(), err)
+ return Error(400, err.Error(), err)
}
- return ApiError(500, "Error while checking dashboard permissions", err)
+ return Error(500, "Error while checking dashboard permissions", err)
}
- return ApiError(403, "Cannot remove own admin permission for a folder", nil)
+ return Error(403, "Cannot remove own admin permission for a folder", nil)
}
if err := bus.Dispatch(&cmd); err != nil {
if err == m.ErrDashboardAclInfoMissing || err == m.ErrDashboardPermissionDashboardEmpty {
- return ApiError(409, err.Error(), err)
+ return Error(409, err.Error(), err)
}
- return ApiError(500, "Failed to create permission", err)
+ return Error(500, "Failed to create permission", err)
}
- return ApiSuccess("Dashboard permissions updated")
+ return Success("Dashboard permissions updated")
}
diff --git a/pkg/api/dashboard_snapshot.go b/pkg/api/dashboard_snapshot.go
index b74dfe5fd34..f474d357df5 100644
--- a/pkg/api/dashboard_snapshot.go
+++ b/pkg/api/dashboard_snapshot.go
@@ -99,11 +99,11 @@ func DeleteDashboardSnapshot(c *m.ReqContext) Response {
err := bus.Dispatch(query)
if err != nil {
- return ApiError(500, "Failed to get dashboard snapshot", err)
+ return Error(500, "Failed to get dashboard snapshot", err)
}
if query.Result == nil {
- return ApiError(404, "Failed to get dashboard snapshot", nil)
+ return Error(404, "Failed to get dashboard snapshot", nil)
}
dashboard := query.Result.Dashboard
dashboardID := dashboard.Get("id").MustInt64()
@@ -111,20 +111,20 @@ func DeleteDashboardSnapshot(c *m.ReqContext) Response {
guardian := guardian.New(dashboardID, c.OrgId, c.SignedInUser)
canEdit, err := guardian.CanEdit()
if err != nil {
- return ApiError(500, "Error while checking permissions for snapshot", err)
+ return Error(500, "Error while checking permissions for snapshot", err)
}
if !canEdit && query.Result.UserId != c.SignedInUser.UserId {
- return ApiError(403, "Access denied to this snapshot", nil)
+ return Error(403, "Access denied to this snapshot", nil)
}
cmd := &m.DeleteDashboardSnapshotCommand{DeleteKey: key}
if err := bus.Dispatch(cmd); err != nil {
- return ApiError(500, "Failed to delete dashboard snapshot", err)
+ return Error(500, "Failed to delete dashboard snapshot", err)
}
- return Json(200, util.DynMap{"message": "Snapshot deleted. It might take an hour before it's cleared from a CDN cache."})
+ return JSON(200, util.DynMap{"message": "Snapshot deleted. It might take an hour before it's cleared from a CDN cache."})
}
// GET /api/dashboard/snapshots
@@ -145,7 +145,7 @@ func SearchDashboardSnapshots(c *m.ReqContext) Response {
err := bus.Dispatch(&searchQuery)
if err != nil {
- return ApiError(500, "Search failed", err)
+ return Error(500, "Search failed", err)
}
dtos := make([]*m.DashboardSnapshotDTO, len(searchQuery.Result))
@@ -165,5 +165,5 @@ func SearchDashboardSnapshots(c *m.ReqContext) Response {
}
}
- return Json(200, dtos)
+ return JSON(200, dtos)
}
diff --git a/pkg/api/dashboard_test.go b/pkg/api/dashboard_test.go
index d940c5406be..0d87023ce40 100644
--- a/pkg/api/dashboard_test.go
+++ b/pkg/api/dashboard_test.go
@@ -638,7 +638,7 @@ func TestDashboardApiEndpoint(t *testing.T) {
Convey("Should result in 412 Precondition failed", func() {
So(sc.resp.Code, ShouldEqual, 412)
- result := sc.ToJson()
+ result := sc.ToJSON()
So(result.Get("status").MustString(), ShouldEqual, "multiple-slugs-exists")
So(result.Get("message").MustString(), ShouldEqual, m.ErrDashboardsWithSameSlugExists.Error())
})
@@ -686,7 +686,7 @@ func TestDashboardApiEndpoint(t *testing.T) {
})
Convey("It should return correct response data", func() {
- result := sc.ToJson()
+ result := sc.ToJSON()
So(result.Get("status").MustString(), ShouldEqual, "success")
So(result.Get("id").MustInt64(), ShouldEqual, 2)
So(result.Get("uid").MustString(), ShouldEqual, "uid")
@@ -903,7 +903,7 @@ func postDiffScenario(desc string, url string, routePattern string, cmd dtos.Cal
})
}
-func (sc *scenarioContext) ToJson() *simplejson.Json {
+func (sc *scenarioContext) ToJSON() *simplejson.Json {
var result *simplejson.Json
err := json.NewDecoder(sc.resp.Body).Decode(&result)
So(err, ShouldBeNil)
diff --git a/pkg/api/dataproxy.go b/pkg/api/dataproxy.go
index 2e97e5d2d6f..33839ca985d 100644
--- a/pkg/api/dataproxy.go
+++ b/pkg/api/dataproxy.go
@@ -13,7 +13,7 @@ import (
const HeaderNameNoBackendCache = "X-Grafana-NoCache"
-func (hs *HttpServer) getDatasourceByID(id int64, orgID int64, nocache bool) (*m.DataSource, error) {
+func (hs *HTTPServer) getDatasourceByID(id int64, orgID int64, nocache bool) (*m.DataSource, error) {
cacheKey := fmt.Sprintf("ds-%d", id)
if !nocache {
@@ -34,7 +34,7 @@ func (hs *HttpServer) getDatasourceByID(id int64, orgID int64, nocache bool) (*m
return query.Result, nil
}
-func (hs *HttpServer) ProxyDataSourceRequest(c *m.ReqContext) {
+func (hs *HTTPServer) ProxyDataSourceRequest(c *m.ReqContext) {
c.TimeRequest(metrics.M_DataSource_ProxyReq_Timer)
nocache := c.Req.Header.Get(HeaderNameNoBackendCache) == "true"
diff --git a/pkg/api/datasources.go b/pkg/api/datasources.go
index 4b5aa56d6e7..99677a93ee6 100644
--- a/pkg/api/datasources.go
+++ b/pkg/api/datasources.go
@@ -14,7 +14,7 @@ func GetDataSources(c *m.ReqContext) Response {
query := m.GetDataSourcesQuery{OrgId: c.OrgId}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Failed to query datasources", err)
+ return Error(500, "Failed to query datasources", err)
}
result := make(dtos.DataSourceList, 0)
@@ -46,7 +46,7 @@ func GetDataSources(c *m.ReqContext) Response {
sort.Sort(result)
- return Json(200, &result)
+ return JSON(200, &result)
}
func GetDataSourceByID(c *m.ReqContext) Response {
@@ -57,66 +57,66 @@ func GetDataSourceByID(c *m.ReqContext) Response {
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrDataSourceNotFound {
- return ApiError(404, "Data source not found", nil)
+ return Error(404, "Data source not found", nil)
}
- return ApiError(500, "Failed to query datasources", err)
+ return Error(500, "Failed to query datasources", err)
}
ds := query.Result
dtos := convertModelToDtos(ds)
- return Json(200, &dtos)
+ return JSON(200, &dtos)
}
func DeleteDataSourceByID(c *m.ReqContext) Response {
id := c.ParamsInt64(":id")
if id <= 0 {
- return ApiError(400, "Missing valid datasource id", nil)
+ return Error(400, "Missing valid datasource id", nil)
}
ds, err := getRawDataSourceByID(id, c.OrgId)
if err != nil {
- return ApiError(400, "Failed to delete datasource", nil)
+ return Error(400, "Failed to delete datasource", nil)
}
if ds.ReadOnly {
- return ApiError(403, "Cannot delete read-only data source", nil)
+ return Error(403, "Cannot delete read-only data source", nil)
}
cmd := &m.DeleteDataSourceByIdCommand{Id: id, OrgId: c.OrgId}
err = bus.Dispatch(cmd)
if err != nil {
- return ApiError(500, "Failed to delete datasource", err)
+ return Error(500, "Failed to delete datasource", err)
}
- return ApiSuccess("Data source deleted")
+ return Success("Data source deleted")
}
func DeleteDataSourceByName(c *m.ReqContext) Response {
name := c.Params(":name")
if name == "" {
- return ApiError(400, "Missing valid datasource name", nil)
+ return Error(400, "Missing valid datasource name", nil)
}
getCmd := &m.GetDataSourceByNameQuery{Name: name, OrgId: c.OrgId}
if err := bus.Dispatch(getCmd); err != nil {
- return ApiError(500, "Failed to delete datasource", err)
+ return Error(500, "Failed to delete datasource", err)
}
if getCmd.Result.ReadOnly {
- return ApiError(403, "Cannot delete read-only data source", nil)
+ return Error(403, "Cannot delete read-only data source", nil)
}
cmd := &m.DeleteDataSourceByNameCommand{Name: name, OrgId: c.OrgId}
err := bus.Dispatch(cmd)
if err != nil {
- return ApiError(500, "Failed to delete datasource", err)
+ return Error(500, "Failed to delete datasource", err)
}
- return ApiSuccess("Data source deleted")
+ return Success("Data source deleted")
}
func AddDataSource(c *m.ReqContext, cmd m.AddDataSourceCommand) Response {
@@ -124,14 +124,14 @@ func AddDataSource(c *m.ReqContext, cmd m.AddDataSourceCommand) Response {
if err := bus.Dispatch(&cmd); err != nil {
if err == m.ErrDataSourceNameExists {
- return ApiError(409, err.Error(), err)
+ return Error(409, err.Error(), err)
}
- return ApiError(500, "Failed to add datasource", err)
+ return Error(500, "Failed to add datasource", err)
}
ds := convertModelToDtos(cmd.Result)
- return Json(200, util.DynMap{
+ return JSON(200, util.DynMap{
"message": "Datasource added",
"id": cmd.Result.Id,
"name": cmd.Result.Name,
@@ -145,18 +145,18 @@ func UpdateDataSource(c *m.ReqContext, cmd m.UpdateDataSourceCommand) Response {
err := fillWithSecureJSONData(&cmd)
if err != nil {
- return ApiError(500, "Failed to update datasource", err)
+ return Error(500, "Failed to update datasource", err)
}
err = bus.Dispatch(&cmd)
if err != nil {
if err == m.ErrDataSourceUpdatingOldVersion {
- return ApiError(500, "Failed to update datasource. Reload new version and try again", err)
+ return Error(500, "Failed to update datasource. Reload new version and try again", err)
}
- return ApiError(500, "Failed to update datasource", err)
+ return Error(500, "Failed to update datasource", err)
}
ds := convertModelToDtos(cmd.Result)
- return Json(200, util.DynMap{
+ return JSON(200, util.DynMap{
"message": "Datasource updated",
"id": cmd.Id,
"name": cmd.Name,
@@ -208,14 +208,14 @@ func GetDataSourceByName(c *m.ReqContext) Response {
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrDataSourceNotFound {
- return ApiError(404, "Data source not found", nil)
+ return Error(404, "Data source not found", nil)
}
- return ApiError(500, "Failed to query datasources", err)
+ return Error(500, "Failed to query datasources", err)
}
dtos := convertModelToDtos(query.Result)
dtos.ReadOnly = true
- return Json(200, &dtos)
+ return JSON(200, &dtos)
}
// Get /api/datasources/id/:name
@@ -224,9 +224,9 @@ func GetDataSourceIDByName(c *m.ReqContext) Response {
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrDataSourceNotFound {
- return ApiError(404, "Data source not found", nil)
+ return Error(404, "Data source not found", nil)
}
- return ApiError(500, "Failed to query datasources", err)
+ return Error(500, "Failed to query datasources", err)
}
ds := query.Result
@@ -234,7 +234,7 @@ func GetDataSourceIDByName(c *m.ReqContext) Response {
Id: ds.Id,
}
- return Json(200, &dtos)
+ return JSON(200, &dtos)
}
func convertModelToDtos(ds *m.DataSource) dtos.DataSource {
diff --git a/pkg/api/dtos/prefs.go b/pkg/api/dtos/prefs.go
index 97e20bb4011..ef1ea4040e5 100644
--- a/pkg/api/dtos/prefs.go
+++ b/pkg/api/dtos/prefs.go
@@ -2,12 +2,12 @@ package dtos
type Prefs struct {
Theme string `json:"theme"`
- HomeDashboardId int64 `json:"homeDashboardId"`
+ HomeDashboardID int64 `json:"homeDashboardId"`
Timezone string `json:"timezone"`
}
type UpdatePrefsCmd struct {
Theme string `json:"theme"`
- HomeDashboardId int64 `json:"homeDashboardId"`
+ HomeDashboardID int64 `json:"homeDashboardId"`
Timezone string `json:"timezone"`
}
diff --git a/pkg/api/folder.go b/pkg/api/folder.go
index e88f7fc2c3b..f0cdff24d20 100644
--- a/pkg/api/folder.go
+++ b/pkg/api/folder.go
@@ -28,30 +28,30 @@ func GetFolders(c *m.ReqContext) Response {
})
}
- return Json(200, result)
+ return JSON(200, result)
}
func GetFolderByUID(c *m.ReqContext) Response {
s := dashboards.NewFolderService(c.OrgId, c.SignedInUser)
- folder, err := s.GetFolderByUid(c.Params(":uid"))
+ folder, err := s.GetFolderByUID(c.Params(":uid"))
if err != nil {
return toFolderError(err)
}
g := guardian.New(folder.Id, c.OrgId, c.SignedInUser)
- return Json(200, toFolderDto(g, folder))
+ return JSON(200, toFolderDto(g, folder))
}
func GetFolderByID(c *m.ReqContext) Response {
s := dashboards.NewFolderService(c.OrgId, c.SignedInUser)
- folder, err := s.GetFolderById(c.ParamsInt64(":id"))
+ folder, err := s.GetFolderByID(c.ParamsInt64(":id"))
if err != nil {
return toFolderError(err)
}
g := guardian.New(folder.Id, c.OrgId, c.SignedInUser)
- return Json(200, toFolderDto(g, folder))
+ return JSON(200, toFolderDto(g, folder))
}
func CreateFolder(c *m.ReqContext, cmd m.CreateFolderCommand) Response {
@@ -62,7 +62,7 @@ func CreateFolder(c *m.ReqContext, cmd m.CreateFolderCommand) Response {
}
g := guardian.New(cmd.Result.Id, c.OrgId, c.SignedInUser)
- return Json(200, toFolderDto(g, cmd.Result))
+ return JSON(200, toFolderDto(g, cmd.Result))
}
func UpdateFolder(c *m.ReqContext, cmd m.UpdateFolderCommand) Response {
@@ -73,7 +73,7 @@ func UpdateFolder(c *m.ReqContext, cmd m.UpdateFolderCommand) Response {
}
g := guardian.New(cmd.Result.Id, c.OrgId, c.SignedInUser)
- return Json(200, toFolderDto(g, cmd.Result))
+ return JSON(200, toFolderDto(g, cmd.Result))
}
func DeleteFolder(c *m.ReqContext) Response {
@@ -83,7 +83,7 @@ func DeleteFolder(c *m.ReqContext) Response {
return toFolderError(err)
}
- return Json(200, util.DynMap{
+ return JSON(200, util.DynMap{
"title": f.Title,
"message": fmt.Sprintf("Folder %s deleted", f.Title),
})
@@ -127,20 +127,20 @@ func toFolderError(err error) Response {
err == m.ErrDashboardTypeMismatch ||
err == m.ErrDashboardInvalidUid ||
err == m.ErrDashboardUidToLong {
- return ApiError(400, err.Error(), nil)
+ return Error(400, err.Error(), nil)
}
if err == m.ErrFolderAccessDenied {
- return ApiError(403, "Access denied", err)
+ return Error(403, "Access denied", err)
}
if err == m.ErrFolderNotFound {
- return Json(404, util.DynMap{"status": "not-found", "message": m.ErrFolderNotFound.Error()})
+ return JSON(404, util.DynMap{"status": "not-found", "message": m.ErrFolderNotFound.Error()})
}
if err == m.ErrFolderVersionMismatch {
- return Json(412, util.DynMap{"status": "version-mismatch", "message": m.ErrFolderVersionMismatch.Error()})
+ return JSON(412, util.DynMap{"status": "version-mismatch", "message": m.ErrFolderVersionMismatch.Error()})
}
- return ApiError(500, "Folder API error", err)
+ return Error(500, "Folder API error", err)
}
diff --git a/pkg/api/folder_permission.go b/pkg/api/folder_permission.go
index 1b04eb20e53..0d0904c99ea 100644
--- a/pkg/api/folder_permission.go
+++ b/pkg/api/folder_permission.go
@@ -12,7 +12,7 @@ import (
func GetFolderPermissionList(c *m.ReqContext) Response {
s := dashboards.NewFolderService(c.OrgId, c.SignedInUser)
- folder, err := s.GetFolderByUid(c.Params(":uid"))
+ folder, err := s.GetFolderByUID(c.Params(":uid"))
if err != nil {
return toFolderError(err)
@@ -26,7 +26,7 @@ func GetFolderPermissionList(c *m.ReqContext) Response {
acl, err := g.GetAcl()
if err != nil {
- return ApiError(500, "Failed to get folder permissions", err)
+ return Error(500, "Failed to get folder permissions", err)
}
for _, perm := range acl {
@@ -38,12 +38,12 @@ func GetFolderPermissionList(c *m.ReqContext) Response {
}
}
- return Json(200, acl)
+ return JSON(200, acl)
}
func UpdateFolderPermissions(c *m.ReqContext, apiCmd dtos.UpdateDashboardAclCommand) Response {
s := dashboards.NewFolderService(c.OrgId, c.SignedInUser)
- folder, err := s.GetFolderByUid(c.Params(":uid"))
+ folder, err := s.GetFolderByUID(c.Params(":uid"))
if err != nil {
return toFolderError(err)
@@ -79,13 +79,13 @@ func UpdateFolderPermissions(c *m.ReqContext, apiCmd dtos.UpdateDashboardAclComm
if err != nil {
if err == guardian.ErrGuardianPermissionExists ||
err == guardian.ErrGuardianOverride {
- return ApiError(400, err.Error(), err)
+ return Error(400, err.Error(), err)
}
- return ApiError(500, "Error while checking folder permissions", err)
+ return Error(500, "Error while checking folder permissions", err)
}
- return ApiError(403, "Cannot remove own admin permission for a folder", nil)
+ return Error(403, "Cannot remove own admin permission for a folder", nil)
}
if err := bus.Dispatch(&cmd); err != nil {
@@ -97,11 +97,11 @@ func UpdateFolderPermissions(c *m.ReqContext, apiCmd dtos.UpdateDashboardAclComm
}
if err == m.ErrFolderAclInfoMissing || err == m.ErrFolderPermissionFolderEmpty {
- return ApiError(409, err.Error(), err)
+ return Error(409, err.Error(), err)
}
- return ApiError(500, "Failed to create permission", err)
+ return Error(500, "Failed to create permission", err)
}
- return ApiSuccess("Folder permissions updated")
+ return Success("Folder permissions updated")
}
diff --git a/pkg/api/folder_permission_test.go b/pkg/api/folder_permission_test.go
index 00d025fdce2..f7458af6dce 100644
--- a/pkg/api/folder_permission_test.go
+++ b/pkg/api/folder_permission_test.go
@@ -17,7 +17,7 @@ func TestFolderPermissionApiEndpoint(t *testing.T) {
Convey("Folder permissions test", t, func() {
Convey("Given folder not exists", func() {
mock := &fakeFolderService{
- GetFolderByUidError: m.ErrFolderNotFound,
+ GetFolderByUIDError: m.ErrFolderNotFound,
}
origNewFolderService := dashboards.NewFolderService
@@ -49,7 +49,7 @@ func TestFolderPermissionApiEndpoint(t *testing.T) {
guardian.MockDashboardGuardian(&guardian.FakeDashboardGuardian{CanAdminValue: false})
mock := &fakeFolderService{
- GetFolderByUidResult: &m.Folder{
+ GetFolderByUIDResult: &m.Folder{
Id: 1,
Uid: "uid",
Title: "Folder",
@@ -96,7 +96,7 @@ func TestFolderPermissionApiEndpoint(t *testing.T) {
})
mock := &fakeFolderService{
- GetFolderByUidResult: &m.Folder{
+ GetFolderByUIDResult: &m.Folder{
Id: 1,
Uid: "uid",
Title: "Folder",
@@ -142,7 +142,7 @@ func TestFolderPermissionApiEndpoint(t *testing.T) {
})
mock := &fakeFolderService{
- GetFolderByUidResult: &m.Folder{
+ GetFolderByUIDResult: &m.Folder{
Id: 1,
Uid: "uid",
Title: "Folder",
@@ -178,7 +178,7 @@ func TestFolderPermissionApiEndpoint(t *testing.T) {
)
mock := &fakeFolderService{
- GetFolderByUidResult: &m.Folder{
+ GetFolderByUIDResult: &m.Folder{
Id: 1,
Uid: "uid",
Title: "Folder",
diff --git a/pkg/api/folder_test.go b/pkg/api/folder_test.go
index e3c63ac0745..0d9b9495686 100644
--- a/pkg/api/folder_test.go
+++ b/pkg/api/folder_test.go
@@ -204,10 +204,10 @@ func updateFolderScenario(desc string, url string, routePattern string, mock *fa
type fakeFolderService struct {
GetFoldersResult []*m.Folder
GetFoldersError error
- GetFolderByUidResult *m.Folder
- GetFolderByUidError error
- GetFolderByIdResult *m.Folder
- GetFolderByIdError error
+ GetFolderByUIDResult *m.Folder
+ GetFolderByUIDError error
+ GetFolderByIDResult *m.Folder
+ GetFolderByIDError error
CreateFolderResult *m.Folder
CreateFolderError error
UpdateFolderResult *m.Folder
@@ -221,12 +221,12 @@ func (s *fakeFolderService) GetFolders(limit int) ([]*m.Folder, error) {
return s.GetFoldersResult, s.GetFoldersError
}
-func (s *fakeFolderService) GetFolderById(id int64) (*m.Folder, error) {
- return s.GetFolderByIdResult, s.GetFolderByIdError
+func (s *fakeFolderService) GetFolderByID(id int64) (*m.Folder, error) {
+ return s.GetFolderByIDResult, s.GetFolderByIDError
}
-func (s *fakeFolderService) GetFolderByUid(uid string) (*m.Folder, error) {
- return s.GetFolderByUidResult, s.GetFolderByUidError
+func (s *fakeFolderService) GetFolderByUID(uid string) (*m.Folder, error) {
+ return s.GetFolderByUIDResult, s.GetFolderByUIDError
}
func (s *fakeFolderService) CreateFolder(cmd *m.CreateFolderCommand) error {
@@ -234,7 +234,7 @@ func (s *fakeFolderService) CreateFolder(cmd *m.CreateFolderCommand) error {
return s.CreateFolderError
}
-func (s *fakeFolderService) UpdateFolder(existingUid string, cmd *m.UpdateFolderCommand) error {
+func (s *fakeFolderService) UpdateFolder(existingUID string, cmd *m.UpdateFolderCommand) error {
cmd.Result = s.UpdateFolderResult
return s.UpdateFolderError
}
diff --git a/pkg/api/http_server.go b/pkg/api/http_server.go
index 772a27bd4cd..921a641b5da 100644
--- a/pkg/api/http_server.go
+++ b/pkg/api/http_server.go
@@ -29,7 +29,7 @@ import (
"github.com/grafana/grafana/pkg/setting"
)
-type HttpServer struct {
+type HTTPServer struct {
log log.Logger
macaron *macaron.Macaron
context context.Context
@@ -39,14 +39,14 @@ type HttpServer struct {
httpSrv *http.Server
}
-func NewHTTPServer() *HttpServer {
- return &HttpServer{
+func NewHTTPServer() *HTTPServer {
+ return &HTTPServer{
log: log.New("http.server"),
cache: gocache.New(5*time.Minute, 10*time.Minute),
}
}
-func (hs *HttpServer) Start(ctx context.Context) error {
+func (hs *HTTPServer) Start(ctx context.Context) error {
var err error
hs.context = ctx
@@ -93,13 +93,13 @@ func (hs *HttpServer) Start(ctx context.Context) error {
return err
}
-func (hs *HttpServer) Shutdown(ctx context.Context) error {
+func (hs *HTTPServer) Shutdown(ctx context.Context) error {
err := hs.httpSrv.Shutdown(ctx)
hs.log.Info("Stopped HTTP server")
return err
}
-func (hs *HttpServer) listenAndServeTLS(certfile, keyfile string) error {
+func (hs *HTTPServer) listenAndServeTLS(certfile, keyfile string) error {
if certfile == "" {
return fmt.Errorf("cert_file cannot be empty when using HTTPS")
}
@@ -141,7 +141,7 @@ func (hs *HttpServer) listenAndServeTLS(certfile, keyfile string) error {
return hs.httpSrv.ListenAndServeTLS(setting.CertFile, setting.KeyFile)
}
-func (hs *HttpServer) newMacaron() *macaron.Macaron {
+func (hs *HTTPServer) newMacaron() *macaron.Macaron {
macaron.Env = setting.Env
m := macaron.New()
@@ -188,7 +188,7 @@ func (hs *HttpServer) newMacaron() *macaron.Macaron {
return m
}
-func (hs *HttpServer) metricsEndpoint(ctx *macaron.Context) {
+func (hs *HTTPServer) metricsEndpoint(ctx *macaron.Context) {
if ctx.Req.Method != "GET" || ctx.Req.URL.Path != "/metrics" {
return
}
@@ -197,7 +197,7 @@ func (hs *HttpServer) metricsEndpoint(ctx *macaron.Context) {
ServeHTTP(ctx.Resp, ctx.Req.Request)
}
-func (hs *HttpServer) healthHandler(ctx *macaron.Context) {
+func (hs *HTTPServer) healthHandler(ctx *macaron.Context) {
notHeadOrGet := ctx.Req.Method != http.MethodGet && ctx.Req.Method != http.MethodHead
if notHeadOrGet || ctx.Req.URL.Path != "/api/health" {
return
@@ -221,7 +221,7 @@ func (hs *HttpServer) healthHandler(ctx *macaron.Context) {
ctx.Resp.Write(dataBytes)
}
-func (hs *HttpServer) mapStatic(m *macaron.Macaron, rootDir string, dir string, prefix string) {
+func (hs *HTTPServer) mapStatic(m *macaron.Macaron, rootDir string, dir string, prefix string) {
headers := func(c *macaron.Context) {
c.Resp.Header().Set("Cache-Control", "public, max-age=3600")
}
diff --git a/pkg/api/login.go b/pkg/api/login.go
index dc5ae730721..671e5fb7ecd 100644
--- a/pkg/api/login.go
+++ b/pkg/api/login.go
@@ -98,7 +98,7 @@ func LoginAPIPing(c *m.ReqContext) {
func LoginPost(c *m.ReqContext, cmd dtos.LoginCommand) Response {
if setting.DisableLoginForm {
- return ApiError(401, "Login is disabled", nil)
+ return Error(401, "Login is disabled", nil)
}
authQuery := login.LoginUserQuery{
@@ -109,10 +109,10 @@ func LoginPost(c *m.ReqContext, cmd dtos.LoginCommand) Response {
if err := bus.Dispatch(&authQuery); err != nil {
if err == login.ErrInvalidCredentials || err == login.ErrTooManyLoginAttempts {
- return ApiError(401, "Invalid username or password", err)
+ return Error(401, "Invalid username or password", err)
}
- return ApiError(500, "Error while trying to authenticate user", err)
+ return Error(500, "Error while trying to authenticate user", err)
}
user := authQuery.User
@@ -130,7 +130,7 @@ func LoginPost(c *m.ReqContext, cmd dtos.LoginCommand) Response {
metrics.M_Api_Login_Post.Inc()
- return Json(200, result)
+ return JSON(200, result)
}
func loginUserWithUser(user *m.User, c *m.ReqContext) {
diff --git a/pkg/api/metrics.go b/pkg/api/metrics.go
index 38bb8dc0688..5c06d652b70 100644
--- a/pkg/api/metrics.go
+++ b/pkg/api/metrics.go
@@ -17,17 +17,17 @@ func QueryMetrics(c *m.ReqContext, reqDto dtos.MetricRequest) Response {
timeRange := tsdb.NewTimeRange(reqDto.From, reqDto.To)
if len(reqDto.Queries) == 0 {
- return ApiError(400, "No queries found in query", nil)
+ return Error(400, "No queries found in query", nil)
}
dsID, err := reqDto.Queries[0].Get("datasourceId").Int64()
if err != nil {
- return ApiError(400, "Query missing datasourceId", nil)
+ return Error(400, "Query missing datasourceId", nil)
}
dsQuery := m.GetDataSourceByIdQuery{Id: dsID, OrgId: c.OrgId}
if err := bus.Dispatch(&dsQuery); err != nil {
- return ApiError(500, "failed to fetch data source", err)
+ return Error(500, "failed to fetch data source", err)
}
request := &tsdb.TsdbQuery{TimeRange: timeRange}
@@ -44,7 +44,7 @@ func QueryMetrics(c *m.ReqContext, reqDto dtos.MetricRequest) Response {
resp, err := tsdb.HandleRequest(context.Background(), dsQuery.Result, request)
if err != nil {
- return ApiError(500, "Metric request error", err)
+ return Error(500, "Metric request error", err)
}
statusCode := 200
@@ -56,7 +56,7 @@ func QueryMetrics(c *m.ReqContext, reqDto dtos.MetricRequest) Response {
}
}
- return Json(statusCode, &resp)
+ return JSON(statusCode, &resp)
}
// GET /api/tsdb/testdata/scenarios
@@ -72,22 +72,22 @@ func GetTestDataScenarios(c *m.ReqContext) Response {
})
}
- return Json(200, &result)
+ return JSON(200, &result)
}
// Genereates a index out of range error
func GenerateError(c *m.ReqContext) Response {
var array []string
- return Json(200, array[20])
+ return JSON(200, array[20])
}
// GET /api/tsdb/testdata/gensql
func GenerateSQLTestData(c *m.ReqContext) Response {
if err := bus.Dispatch(&m.InsertSqlTestDataCommand{}); err != nil {
- return ApiError(500, "Failed to insert test data", err)
+ return Error(500, "Failed to insert test data", err)
}
- return Json(200, &util.DynMap{"message": "OK"})
+ return JSON(200, &util.DynMap{"message": "OK"})
}
// GET /api/tsdb/testdata/random-walk
@@ -111,8 +111,8 @@ func GetTestDataRandomWalk(c *m.ReqContext) Response {
resp, err := tsdb.HandleRequest(context.Background(), dsInfo, request)
if err != nil {
- return ApiError(500, "Metric request error", err)
+ return Error(500, "Metric request error", err)
}
- return Json(200, &resp)
+ return JSON(200, &resp)
}
diff --git a/pkg/api/org.go b/pkg/api/org.go
index 7735bd6a7eb..8d0d48bad13 100644
--- a/pkg/api/org.go
+++ b/pkg/api/org.go
@@ -24,10 +24,10 @@ func GetOrgByName(c *m.ReqContext) Response {
query := m.GetOrgByNameQuery{Name: c.Params(":name")}
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrOrgNotFound {
- return ApiError(404, "Organization not found", err)
+ return Error(404, "Organization not found", err)
}
- return ApiError(500, "Failed to get organization", err)
+ return Error(500, "Failed to get organization", err)
}
org := query.Result
result := m.OrgDetailsDTO{
@@ -43,18 +43,18 @@ func GetOrgByName(c *m.ReqContext) Response {
},
}
- return Json(200, &result)
+ return JSON(200, &result)
}
-func getOrgHelper(orgId int64) Response {
- query := m.GetOrgByIdQuery{Id: orgId}
+func getOrgHelper(orgID int64) Response {
+ query := m.GetOrgByIdQuery{Id: orgID}
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrOrgNotFound {
- return ApiError(404, "Organization not found", err)
+ return Error(404, "Organization not found", err)
}
- return ApiError(500, "Failed to get organization", err)
+ return Error(500, "Failed to get organization", err)
}
org := query.Result
@@ -71,26 +71,26 @@ func getOrgHelper(orgId int64) Response {
},
}
- return Json(200, &result)
+ return JSON(200, &result)
}
// POST /api/orgs
func CreateOrg(c *m.ReqContext, cmd m.CreateOrgCommand) Response {
if !c.IsSignedIn || (!setting.AllowUserOrgCreate && !c.IsGrafanaAdmin) {
- return ApiError(403, "Access denied", nil)
+ return Error(403, "Access denied", nil)
}
cmd.UserId = c.UserId
if err := bus.Dispatch(&cmd); err != nil {
if err == m.ErrOrgNameTaken {
- return ApiError(409, "Organization name taken", err)
+ return Error(409, "Organization name taken", err)
}
- return ApiError(500, "Failed to create organization", err)
+ return Error(500, "Failed to create organization", err)
}
metrics.M_Api_Org_Create.Inc()
- return Json(200, &util.DynMap{
+ return JSON(200, &util.DynMap{
"orgId": cmd.Result.Id,
"message": "Organization created",
})
@@ -110,12 +110,12 @@ func updateOrgHelper(form dtos.UpdateOrgForm, orgID int64) Response {
cmd := m.UpdateOrgCommand{Name: form.Name, OrgId: orgID}
if err := bus.Dispatch(&cmd); err != nil {
if err == m.ErrOrgNameTaken {
- return ApiError(400, "Organization name taken", err)
+ return Error(400, "Organization name taken", err)
}
- return ApiError(500, "Failed to update organization", err)
+ return Error(500, "Failed to update organization", err)
}
- return ApiSuccess("Organization updated")
+ return Success("Organization updated")
}
// PUT /api/org/address
@@ -142,21 +142,21 @@ func updateOrgAddressHelper(form dtos.UpdateOrgAddressForm, orgID int64) Respons
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to update org address", err)
+ return Error(500, "Failed to update org address", err)
}
- return ApiSuccess("Address updated")
+ return Success("Address updated")
}
// GET /api/orgs/:orgId
func DeleteOrgByID(c *m.ReqContext) Response {
if err := bus.Dispatch(&m.DeleteOrgCommand{Id: c.ParamsInt64(":orgId")}); err != nil {
if err == m.ErrOrgNotFound {
- return ApiError(404, "Failed to delete organization. ID not found", nil)
+ return Error(404, "Failed to delete organization. ID not found", nil)
}
- return ApiError(500, "Failed to update organization", err)
+ return Error(500, "Failed to update organization", err)
}
- return ApiSuccess("Organization deleted")
+ return Success("Organization deleted")
}
func SearchOrgs(c *m.ReqContext) Response {
@@ -168,8 +168,8 @@ func SearchOrgs(c *m.ReqContext) Response {
}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Failed to search orgs", err)
+ return Error(500, "Failed to search orgs", err)
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
diff --git a/pkg/api/org_invite.go b/pkg/api/org_invite.go
index 0486287a31b..d6ab1c9d372 100644
--- a/pkg/api/org_invite.go
+++ b/pkg/api/org_invite.go
@@ -16,30 +16,30 @@ func GetPendingOrgInvites(c *m.ReqContext) Response {
query := m.GetTempUsersQuery{OrgId: c.OrgId, Status: m.TmpUserInvitePending}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Failed to get invites from db", err)
+ return Error(500, "Failed to get invites from db", err)
}
for _, invite := range query.Result {
invite.Url = setting.ToAbsUrl("invite/" + invite.Code)
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
func AddOrgInvite(c *m.ReqContext, inviteDto dtos.AddInviteForm) Response {
if !inviteDto.Role.IsValid() {
- return ApiError(400, "Invalid role specified", nil)
+ return Error(400, "Invalid role specified", nil)
}
// first try get existing user
userQuery := m.GetUserByLoginQuery{LoginOrEmail: inviteDto.LoginOrEmail}
if err := bus.Dispatch(&userQuery); err != nil {
if err != m.ErrUserNotFound {
- return ApiError(500, "Failed to query db for existing user check", err)
+ return Error(500, "Failed to query db for existing user check", err)
}
if setting.DisableLoginForm {
- return ApiError(401, "User could not be found", nil)
+ return Error(401, "User could not be found", nil)
}
} else {
return inviteExistingUserToOrg(c, userQuery.Result, &inviteDto)
@@ -56,7 +56,7 @@ func AddOrgInvite(c *m.ReqContext, inviteDto dtos.AddInviteForm) Response {
cmd.RemoteAddr = c.Req.RemoteAddr
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to save invite to database", err)
+ return Error(500, "Failed to save invite to database", err)
}
// send invite email
@@ -74,18 +74,18 @@ func AddOrgInvite(c *m.ReqContext, inviteDto dtos.AddInviteForm) Response {
}
if err := bus.Dispatch(&emailCmd); err != nil {
- return ApiError(500, "Failed to send email invite", err)
+ return Error(500, "Failed to send email invite", err)
}
emailSentCmd := m.UpdateTempUserWithEmailSentCommand{Code: cmd.Result.Code}
if err := bus.Dispatch(&emailSentCmd); err != nil {
- return ApiError(500, "Failed to update invite with email sent info", err)
+ return Error(500, "Failed to update invite with email sent info", err)
}
- return ApiSuccess(fmt.Sprintf("Sent invite to %s", inviteDto.LoginOrEmail))
+ return Success(fmt.Sprintf("Sent invite to %s", inviteDto.LoginOrEmail))
}
- return ApiSuccess(fmt.Sprintf("Created invite for %s", inviteDto.LoginOrEmail))
+ return Success(fmt.Sprintf("Created invite for %s", inviteDto.LoginOrEmail))
}
func inviteExistingUserToOrg(c *m.ReqContext, user *m.User, inviteDto *dtos.AddInviteForm) Response {
@@ -93,9 +93,9 @@ func inviteExistingUserToOrg(c *m.ReqContext, user *m.User, inviteDto *dtos.AddI
createOrgUserCmd := m.AddOrgUserCommand{OrgId: c.OrgId, UserId: user.Id, Role: inviteDto.Role}
if err := bus.Dispatch(&createOrgUserCmd); err != nil {
if err == m.ErrOrgUserAlreadyAdded {
- return ApiError(412, fmt.Sprintf("User %s is already added to organization", inviteDto.LoginOrEmail), err)
+ return Error(412, fmt.Sprintf("User %s is already added to organization", inviteDto.LoginOrEmail), err)
}
- return ApiError(500, "Error while trying to create org user", err)
+ return Error(500, "Error while trying to create org user", err)
}
if inviteDto.SendEmail && util.IsEmail(user.Email) {
@@ -110,11 +110,11 @@ func inviteExistingUserToOrg(c *m.ReqContext, user *m.User, inviteDto *dtos.AddI
}
if err := bus.Dispatch(&emailCmd); err != nil {
- return ApiError(500, "Failed to send email invited_to_org", err)
+ return Error(500, "Failed to send email invited_to_org", err)
}
}
- return ApiSuccess(fmt.Sprintf("Existing Grafana user %s added to org %s", user.NameOrFallback(), c.OrgName))
+ return Success(fmt.Sprintf("Existing Grafana user %s added to org %s", user.NameOrFallback(), c.OrgName))
}
func RevokeInvite(c *m.ReqContext) Response {
@@ -122,7 +122,7 @@ func RevokeInvite(c *m.ReqContext) Response {
return rsp
}
- return ApiSuccess("Invite revoked")
+ return Success("Invite revoked")
}
func GetInviteInfoByCode(c *m.ReqContext) Response {
@@ -130,14 +130,14 @@ func GetInviteInfoByCode(c *m.ReqContext) Response {
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrTempUserNotFound {
- return ApiError(404, "Invite not found", nil)
+ return Error(404, "Invite not found", nil)
}
- return ApiError(500, "Failed to get invite", err)
+ return Error(500, "Failed to get invite", err)
}
invite := query.Result
- return Json(200, dtos.InviteInfo{
+ return JSON(200, dtos.InviteInfo{
Email: invite.Email,
Name: invite.Name,
Username: invite.Email,
@@ -150,14 +150,14 @@ func CompleteInvite(c *m.ReqContext, completeInvite dtos.CompleteInviteForm) Res
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrTempUserNotFound {
- return ApiError(404, "Invite not found", nil)
+ return Error(404, "Invite not found", nil)
}
- return ApiError(500, "Failed to get invite", err)
+ return Error(500, "Failed to get invite", err)
}
invite := query.Result
if invite.Status != m.TmpUserInvitePending {
- return ApiError(412, fmt.Sprintf("Invite cannot be used in status %s", invite.Status), nil)
+ return Error(412, fmt.Sprintf("Invite cannot be used in status %s", invite.Status), nil)
}
cmd := m.CreateUserCommand{
@@ -169,7 +169,7 @@ func CompleteInvite(c *m.ReqContext, completeInvite dtos.CompleteInviteForm) Res
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "failed to create user", err)
+ return Error(500, "failed to create user", err)
}
user := &cmd.Result
@@ -188,14 +188,14 @@ func CompleteInvite(c *m.ReqContext, completeInvite dtos.CompleteInviteForm) Res
metrics.M_Api_User_SignUpCompleted.Inc()
metrics.M_Api_User_SignUpInvite.Inc()
- return ApiSuccess("User created and logged in")
+ return Success("User created and logged in")
}
func updateTempUserStatus(code string, status m.TempUserStatus) (bool, Response) {
// update temp user status
updateTmpUserCmd := m.UpdateTempUserStatusCommand{Code: code, Status: status}
if err := bus.Dispatch(&updateTmpUserCmd); err != nil {
- return false, ApiError(500, "Failed to update invite status", err)
+ return false, Error(500, "Failed to update invite status", err)
}
return true, nil
@@ -206,7 +206,7 @@ func applyUserInvite(user *m.User, invite *m.TempUserDTO, setActive bool) (bool,
addOrgUserCmd := m.AddOrgUserCommand{OrgId: invite.OrgId, UserId: user.Id, Role: invite.Role}
if err := bus.Dispatch(&addOrgUserCmd); err != nil {
if err != m.ErrOrgUserAlreadyAdded {
- return false, ApiError(500, "Error while trying to create org user", err)
+ return false, Error(500, "Error while trying to create org user", err)
}
}
@@ -218,7 +218,7 @@ func applyUserInvite(user *m.User, invite *m.TempUserDTO, setActive bool) (bool,
if setActive {
// set org to active
if err := bus.Dispatch(&m.SetUsingOrgCommand{OrgId: invite.OrgId, UserId: user.Id}); err != nil {
- return false, ApiError(500, "Failed to set org as active", err)
+ return false, Error(500, "Failed to set org as active", err)
}
}
diff --git a/pkg/api/org_users.go b/pkg/api/org_users.go
index cd1430d22fb..4e2ed36431e 100644
--- a/pkg/api/org_users.go
+++ b/pkg/api/org_users.go
@@ -20,13 +20,13 @@ func AddOrgUser(c *m.ReqContext, cmd m.AddOrgUserCommand) Response {
func addOrgUserHelper(cmd m.AddOrgUserCommand) Response {
if !cmd.Role.IsValid() {
- return ApiError(400, "Invalid role specified", nil)
+ return Error(400, "Invalid role specified", nil)
}
userQuery := m.GetUserByLoginQuery{LoginOrEmail: cmd.LoginOrEmail}
err := bus.Dispatch(&userQuery)
if err != nil {
- return ApiError(404, "User not found", nil)
+ return Error(404, "User not found", nil)
}
userToAdd := userQuery.Result
@@ -35,12 +35,12 @@ func addOrgUserHelper(cmd m.AddOrgUserCommand) Response {
if err := bus.Dispatch(&cmd); err != nil {
if err == m.ErrOrgUserAlreadyAdded {
- return ApiError(409, "User is already member of this organization", nil)
+ return Error(409, "User is already member of this organization", nil)
}
- return ApiError(500, "Could not add user to organization", err)
+ return Error(500, "Could not add user to organization", err)
}
- return ApiSuccess("User added to organization")
+ return Success("User added to organization")
}
// GET /api/org/users
@@ -61,14 +61,14 @@ func getOrgUsersHelper(orgID int64, query string, limit int) Response {
}
if err := bus.Dispatch(&q); err != nil {
- return ApiError(500, "Failed to get account user", err)
+ return Error(500, "Failed to get account user", err)
}
for _, user := range q.Result {
user.AvatarUrl = dtos.GetGravatarUrl(user.Email)
}
- return Json(200, q.Result)
+ return JSON(200, q.Result)
}
// PATCH /api/org/users/:userId
@@ -87,17 +87,17 @@ func UpdateOrgUser(c *m.ReqContext, cmd m.UpdateOrgUserCommand) Response {
func updateOrgUserHelper(cmd m.UpdateOrgUserCommand) Response {
if !cmd.Role.IsValid() {
- return ApiError(400, "Invalid role specified", nil)
+ return Error(400, "Invalid role specified", nil)
}
if err := bus.Dispatch(&cmd); err != nil {
if err == m.ErrLastOrgAdmin {
- return ApiError(400, "Cannot change role so that there is no organization admin left", nil)
+ return Error(400, "Cannot change role so that there is no organization admin left", nil)
}
- return ApiError(500, "Failed update org user", err)
+ return Error(500, "Failed update org user", err)
}
- return ApiSuccess("Organization user updated")
+ return Success("Organization user updated")
}
// DELETE /api/org/users/:userId
@@ -118,10 +118,10 @@ func removeOrgUserHelper(orgID int64, userID int64) Response {
if err := bus.Dispatch(&cmd); err != nil {
if err == m.ErrLastOrgAdmin {
- return ApiError(400, "Cannot remove last organization admin", nil)
+ return Error(400, "Cannot remove last organization admin", nil)
}
- return ApiError(500, "Failed to remove user from organization", err)
+ return Error(500, "Failed to remove user from organization", err)
}
- return ApiSuccess("User removed from organization")
+ return Success("User removed from organization")
}
diff --git a/pkg/api/password.go b/pkg/api/password.go
index 31ea5d91b34..7dd901c898e 100644
--- a/pkg/api/password.go
+++ b/pkg/api/password.go
@@ -12,15 +12,15 @@ func SendResetPasswordEmail(c *m.ReqContext, form dtos.SendResetPasswordEmailFor
if err := bus.Dispatch(&userQuery); err != nil {
c.Logger.Info("Requested password reset for user that was not found", "user", userQuery.LoginOrEmail)
- return ApiError(200, "Email sent", err)
+ return Error(200, "Email sent", err)
}
emailCmd := m.SendResetPasswordEmailCommand{User: userQuery.Result}
if err := bus.Dispatch(&emailCmd); err != nil {
- return ApiError(500, "Failed to send email", err)
+ return Error(500, "Failed to send email", err)
}
- return ApiSuccess("Email sent")
+ return Success("Email sent")
}
func ResetPassword(c *m.ReqContext, form dtos.ResetUserPasswordForm) Response {
@@ -28,13 +28,13 @@ func ResetPassword(c *m.ReqContext, form dtos.ResetUserPasswordForm) Response {
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrInvalidEmailCode {
- return ApiError(400, "Invalid or expired reset password code", nil)
+ return Error(400, "Invalid or expired reset password code", nil)
}
- return ApiError(500, "Unknown error validating email code", err)
+ return Error(500, "Unknown error validating email code", err)
}
if form.NewPassword != form.ConfirmPassword {
- return ApiError(400, "Passwords do not match", nil)
+ return Error(400, "Passwords do not match", nil)
}
cmd := m.ChangeUserPasswordCommand{}
@@ -42,8 +42,8 @@ func ResetPassword(c *m.ReqContext, form dtos.ResetUserPasswordForm) Response {
cmd.NewPassword = util.EncodePassword(form.NewPassword, query.Result.Salt)
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to change user password", err)
+ return Error(500, "Failed to change user password", err)
}
- return ApiSuccess("User password changed")
+ return Success("User password changed")
}
diff --git a/pkg/api/playlist.go b/pkg/api/playlist.go
index 0198850252d..d2413dfbb4c 100644
--- a/pkg/api/playlist.go
+++ b/pkg/api/playlist.go
@@ -55,10 +55,10 @@ func SearchPlaylists(c *m.ReqContext) Response {
err := bus.Dispatch(&searchQuery)
if err != nil {
- return ApiError(500, "Search failed", err)
+ return Error(500, "Search failed", err)
}
- return Json(200, searchQuery.Result)
+ return JSON(200, searchQuery.Result)
}
func GetPlaylist(c *m.ReqContext) Response {
@@ -66,7 +66,7 @@ func GetPlaylist(c *m.ReqContext) Response {
cmd := m.GetPlaylistByIdQuery{Id: id}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Playlist not found", err)
+ return Error(500, "Playlist not found", err)
}
playlistDTOs, _ := LoadPlaylistItemDTOs(id)
@@ -79,7 +79,7 @@ func GetPlaylist(c *m.ReqContext) Response {
Items: playlistDTOs,
}
- return Json(200, dto)
+ return JSON(200, dto)
}
func LoadPlaylistItemDTOs(id int64) ([]m.PlaylistItemDTO, error) {
@@ -120,10 +120,10 @@ func GetPlaylistItems(c *m.ReqContext) Response {
playlistDTOs, err := LoadPlaylistItemDTOs(id)
if err != nil {
- return ApiError(500, "Could not load playlist items", err)
+ return Error(500, "Could not load playlist items", err)
}
- return Json(200, playlistDTOs)
+ return JSON(200, playlistDTOs)
}
func GetPlaylistDashboards(c *m.ReqContext) Response {
@@ -131,10 +131,10 @@ func GetPlaylistDashboards(c *m.ReqContext) Response {
playlists, err := LoadPlaylistDashboards(c.OrgId, c.SignedInUser, playlistID)
if err != nil {
- return ApiError(500, "Could not load dashboards", err)
+ return Error(500, "Could not load dashboards", err)
}
- return Json(200, playlists)
+ return JSON(200, playlists)
}
func DeletePlaylist(c *m.ReqContext) Response {
@@ -142,34 +142,34 @@ func DeletePlaylist(c *m.ReqContext) Response {
cmd := m.DeletePlaylistCommand{Id: id, OrgId: c.OrgId}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to delete playlist", err)
+ return Error(500, "Failed to delete playlist", err)
}
- return Json(200, "")
+ return JSON(200, "")
}
func CreatePlaylist(c *m.ReqContext, cmd m.CreatePlaylistCommand) Response {
cmd.OrgId = c.OrgId
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to create playlist", err)
+ return Error(500, "Failed to create playlist", err)
}
- return Json(200, cmd.Result)
+ return JSON(200, cmd.Result)
}
func UpdatePlaylist(c *m.ReqContext, cmd m.UpdatePlaylistCommand) Response {
cmd.OrgId = c.OrgId
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to save playlist", err)
+ return Error(500, "Failed to save playlist", err)
}
playlistDTOs, err := LoadPlaylistItemDTOs(cmd.Id)
if err != nil {
- return ApiError(500, "Failed to save playlist", err)
+ return Error(500, "Failed to save playlist", err)
}
cmd.Result.Items = playlistDTOs
- return Json(200, cmd.Result)
+ return JSON(200, cmd.Result)
}
diff --git a/pkg/api/playlist_play.go b/pkg/api/playlist_play.go
index 69a2caef7e0..e82c7b438b4 100644
--- a/pkg/api/playlist_play.go
+++ b/pkg/api/playlist_play.go
@@ -11,11 +11,11 @@ import (
"github.com/grafana/grafana/pkg/services/search"
)
-func populateDashboardsById(dashboardByIds []int64, dashboardIdOrder map[int64]int) (dtos.PlaylistDashboardsSlice, error) {
+func populateDashboardsByID(dashboardByIDs []int64, dashboardIDOrder map[int64]int) (dtos.PlaylistDashboardsSlice, error) {
result := make(dtos.PlaylistDashboardsSlice, 0)
- if len(dashboardByIds) > 0 {
- dashboardQuery := m.GetDashboardsQuery{DashboardIds: dashboardByIds}
+ if len(dashboardByIDs) > 0 {
+ dashboardQuery := m.GetDashboardsQuery{DashboardIds: dashboardByIDs}
if err := bus.Dispatch(&dashboardQuery); err != nil {
return result, err
}
@@ -26,7 +26,7 @@ func populateDashboardsById(dashboardByIds []int64, dashboardIdOrder map[int64]i
Slug: item.Slug,
Title: item.Title,
Uri: "db/" + item.Slug,
- Order: dashboardIdOrder[item.Id],
+ Order: dashboardIDOrder[item.Id],
})
}
}
@@ -85,7 +85,7 @@ func LoadPlaylistDashboards(orgID int64, signedInUser *m.SignedInUser, playlistI
result := make(dtos.PlaylistDashboardsSlice, 0)
- var k, _ = populateDashboardsById(dashboardByIDs, dashboardIDOrder)
+ var k, _ = populateDashboardsByID(dashboardByIDs, dashboardIDOrder)
result = append(result, k...)
result = append(result, populateDashboardsByTag(orgID, signedInUser, dashboardByTag, dashboardTagOrder)...)
diff --git a/pkg/api/pluginproxy/ds_proxy.go b/pkg/api/pluginproxy/ds_proxy.go
index b861a344c75..7d2d804c007 100644
--- a/pkg/api/pluginproxy/ds_proxy.go
+++ b/pkg/api/pluginproxy/ds_proxy.go
@@ -25,8 +25,8 @@ import (
)
var (
- logger log.Logger = log.New("data-proxy-log")
- client *http.Client = &http.Client{
+ logger = log.New("data-proxy-log")
+ client = &http.Client{
Timeout: time.Second * 30,
Transport: &http.Transport{Proxy: http.ProxyFromEnvironment},
}
@@ -49,14 +49,14 @@ type DataSourceProxy struct {
}
func NewDataSourceProxy(ds *m.DataSource, plugin *plugins.DataSourcePlugin, ctx *m.ReqContext, proxyPath string) *DataSourceProxy {
- targetUrl, _ := url.Parse(ds.Url)
+ targetURL, _ := url.Parse(ds.Url)
return &DataSourceProxy{
ds: ds,
plugin: plugin,
ctx: ctx,
proxyPath: proxyPath,
- targetUrl: targetUrl,
+ targetUrl: targetURL,
}
}
@@ -279,16 +279,16 @@ func (proxy *DataSourceProxy) applyRoute(req *http.Request) {
SecureJsonData: proxy.ds.SecureJsonData.Decrypt(),
}
- routeUrl, err := url.Parse(proxy.route.Url)
+ routeURL, err := url.Parse(proxy.route.Url)
if err != nil {
logger.Error("Error parsing plugin route url")
return
}
- req.URL.Scheme = routeUrl.Scheme
- req.URL.Host = routeUrl.Host
- req.Host = routeUrl.Host
- req.URL.Path = util.JoinUrlFragments(routeUrl.Path, proxy.proxyPath)
+ req.URL.Scheme = routeURL.Scheme
+ req.URL.Host = routeURL.Host
+ req.Host = routeURL.Host
+ req.URL.Path = util.JoinUrlFragments(routeURL.Path, proxy.proxyPath)
if err := addHeaders(&req.Header, proxy.route, data); err != nil {
logger.Error("Failed to render plugin headers", "error", err)
@@ -320,11 +320,11 @@ func (proxy *DataSourceProxy) getAccessToken(data templateData) (string, error)
params := make(url.Values)
for key, value := range proxy.route.TokenAuth.Params {
- if interpolatedParam, err := interpolateString(value, data); err != nil {
+ interpolatedParam, err := interpolateString(value, data)
+ if err != nil {
return "", err
- } else {
- params.Add(key, interpolatedParam)
}
+ params.Add(key, interpolatedParam)
}
getTokenReq, _ := http.NewRequest("POST", urlInterpolated, bytes.NewBufferString(params.Encode()))
@@ -354,13 +354,13 @@ func (proxy *DataSourceProxy) getAccessToken(data templateData) (string, error)
func interpolateString(text string, data templateData) (string, error) {
t, err := template.New("content").Parse(text)
if err != nil {
- return "", errors.New(fmt.Sprintf("Could not parse template %s.", text))
+ return "", fmt.Errorf("could not parse template %s", text)
}
var contentBuf bytes.Buffer
err = t.Execute(&contentBuf, data)
if err != nil {
- return "", errors.New(fmt.Sprintf("Failed to execute template %s.", text))
+ return "", fmt.Errorf("failed to execute template %s", text)
}
return contentBuf.String(), nil
diff --git a/pkg/api/pluginproxy/ds_proxy_test.go b/pkg/api/pluginproxy/ds_proxy_test.go
index 3cf67d9178a..3fc1392a851 100644
--- a/pkg/api/pluginproxy/ds_proxy_test.go
+++ b/pkg/api/pluginproxy/ds_proxy_test.go
@@ -107,8 +107,8 @@ func TestDSRouteRule(t *testing.T) {
proxy := NewDataSourceProxy(ds, plugin, ctx, "/render")
- requestUrl, _ := url.Parse("http://grafana.com/sub")
- req := http.Request{URL: requestUrl}
+ requestURL, _ := url.Parse("http://grafana.com/sub")
+ req := http.Request{URL: requestURL}
proxy.getDirector()(&req)
@@ -132,8 +132,8 @@ func TestDSRouteRule(t *testing.T) {
ctx := &m.ReqContext{}
proxy := NewDataSourceProxy(ds, plugin, ctx, "")
- requestUrl, _ := url.Parse("http://grafana.com/sub")
- req := http.Request{URL: requestUrl}
+ requestURL, _ := url.Parse("http://grafana.com/sub")
+ req := http.Request{URL: requestURL}
proxy.getDirector()(&req)
@@ -162,8 +162,8 @@ func TestDSRouteRule(t *testing.T) {
ctx := &m.ReqContext{}
proxy := NewDataSourceProxy(ds, plugin, ctx, "")
- requestUrl, _ := url.Parse("http://grafana.com/sub")
- req := http.Request{URL: requestUrl, Header: make(http.Header)}
+ requestURL, _ := url.Parse("http://grafana.com/sub")
+ req := http.Request{URL: requestURL, Header: make(http.Header)}
cookies := "grafana_user=admin; grafana_remember=99; grafana_sess=11; JSESSION_ID=test"
req.Header.Set("Cookie", cookies)
@@ -188,8 +188,8 @@ func TestDSRouteRule(t *testing.T) {
ctx := &m.ReqContext{}
proxy := NewDataSourceProxy(ds, plugin, ctx, "")
- requestUrl, _ := url.Parse("http://grafana.com/sub")
- req := http.Request{URL: requestUrl, Header: make(http.Header)}
+ requestURL, _ := url.Parse("http://grafana.com/sub")
+ req := http.Request{URL: requestURL, Header: make(http.Header)}
cookies := "grafana_user=admin; grafana_remember=99; grafana_sess=11; JSESSION_ID=test"
req.Header.Set("Cookie", cookies)
diff --git a/pkg/api/pluginproxy/pluginproxy.go b/pkg/api/pluginproxy/pluginproxy.go
index eb78250838a..ffbe470cb20 100644
--- a/pkg/api/pluginproxy/pluginproxy.go
+++ b/pkg/api/pluginproxy/pluginproxy.go
@@ -19,10 +19,10 @@ type templateData struct {
SecureJsonData map[string]string
}
-func getHeaders(route *plugins.AppPluginRoute, orgId int64, appId string) (http.Header, error) {
+func getHeaders(route *plugins.AppPluginRoute, orgId int64, appID string) (http.Header, error) {
result := http.Header{}
- query := m.GetPluginSettingByIdQuery{OrgId: orgId, PluginId: appId}
+ query := m.GetPluginSettingByIdQuery{OrgId: orgId, PluginId: appID}
if err := bus.Dispatch(&query); err != nil {
return nil, err
@@ -37,16 +37,16 @@ func getHeaders(route *plugins.AppPluginRoute, orgId int64, appId string) (http.
return result, err
}
-func NewApiPluginProxy(ctx *m.ReqContext, proxyPath string, route *plugins.AppPluginRoute, appId string) *httputil.ReverseProxy {
- targetUrl, _ := url.Parse(route.Url)
+func NewApiPluginProxy(ctx *m.ReqContext, proxyPath string, route *plugins.AppPluginRoute, appID string) *httputil.ReverseProxy {
+ targetURL, _ := url.Parse(route.Url)
director := func(req *http.Request) {
- req.URL.Scheme = targetUrl.Scheme
- req.URL.Host = targetUrl.Host
- req.Host = targetUrl.Host
+ req.URL.Scheme = targetURL.Scheme
+ req.URL.Host = targetURL.Host
+ req.Host = targetURL.Host
- req.URL.Path = util.JoinUrlFragments(targetUrl.Path, proxyPath)
+ req.URL.Path = util.JoinUrlFragments(targetURL.Path, proxyPath)
// clear cookie headers
req.Header.Del("Cookie")
@@ -80,7 +80,7 @@ func NewApiPluginProxy(ctx *m.ReqContext, proxyPath string, route *plugins.AppPl
req.Header.Add("X-Grafana-Context", string(ctxJson))
if len(route.Headers) > 0 {
- headers, err := getHeaders(route, ctx.OrgId, appId)
+ headers, err := getHeaders(route, ctx.OrgId, appID)
if err != nil {
ctx.JsonApiErr(500, "Could not generate plugin route header", err)
return
diff --git a/pkg/api/plugins.go b/pkg/api/plugins.go
index 81b3ac10cef..f757f2b9adc 100644
--- a/pkg/api/plugins.go
+++ b/pkg/api/plugins.go
@@ -19,7 +19,7 @@ func GetPluginList(c *m.ReqContext) Response {
pluginSettingsMap, err := plugins.GetPluginSettings(c.OrgId)
if err != nil {
- return ApiError(500, "Failed to get list of plugins", err)
+ return Error(500, "Failed to get list of plugins", err)
}
result := make(dtos.PluginList, 0)
@@ -75,7 +75,7 @@ func GetPluginList(c *m.ReqContext) Response {
}
sort.Sort(result)
- return Json(200, result)
+ return JSON(200, result)
}
func GetPluginSettingByID(c *m.ReqContext) Response {
@@ -83,7 +83,7 @@ func GetPluginSettingByID(c *m.ReqContext) Response {
def, exists := plugins.Plugins[pluginID]
if !exists {
- return ApiError(404, "Plugin not found, no installed plugin with that id", nil)
+ return Error(404, "Plugin not found, no installed plugin with that id", nil)
}
dto := &dtos.PluginSetting{
@@ -104,7 +104,7 @@ func GetPluginSettingByID(c *m.ReqContext) Response {
query := m.GetPluginSettingByIdQuery{PluginId: pluginID, OrgId: c.OrgId}
if err := bus.Dispatch(&query); err != nil {
if err != m.ErrPluginSettingNotFound {
- return ApiError(500, "Failed to get login settings", nil)
+ return Error(500, "Failed to get login settings", nil)
}
} else {
dto.Enabled = query.Result.Enabled
@@ -112,7 +112,7 @@ func GetPluginSettingByID(c *m.ReqContext) Response {
dto.JsonData = query.Result.JsonData
}
- return Json(200, dto)
+ return JSON(200, dto)
}
func UpdatePluginSetting(c *m.ReqContext, cmd m.UpdatePluginSettingCmd) Response {
@@ -122,14 +122,14 @@ func UpdatePluginSetting(c *m.ReqContext, cmd m.UpdatePluginSettingCmd) Response
cmd.PluginId = pluginID
if _, ok := plugins.Apps[cmd.PluginId]; !ok {
- return ApiError(404, "Plugin not installed.", nil)
+ return Error(404, "Plugin not installed.", nil)
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to update plugin setting", err)
+ return Error(500, "Failed to update plugin setting", err)
}
- return ApiSuccess("Plugin settings updated")
+ return Success("Plugin settings updated")
}
func GetPluginDashboards(c *m.ReqContext) Response {
@@ -138,13 +138,13 @@ func GetPluginDashboards(c *m.ReqContext) Response {
list, err := plugins.GetPluginDashboards(c.OrgId, pluginID)
if err != nil {
if notfound, ok := err.(plugins.PluginNotFoundError); ok {
- return ApiError(404, notfound.Error(), nil)
+ return Error(404, notfound.Error(), nil)
}
- return ApiError(500, "Failed to get plugin dashboards", err)
+ return Error(500, "Failed to get plugin dashboards", err)
}
- return Json(200, list)
+ return JSON(200, list)
}
func GetPluginMarkdown(c *m.ReqContext) Response {
@@ -154,10 +154,10 @@ func GetPluginMarkdown(c *m.ReqContext) Response {
content, err := plugins.GetPluginMarkdown(pluginID, name)
if err != nil {
if notfound, ok := err.(plugins.PluginNotFoundError); ok {
- return ApiError(404, notfound.Error(), nil)
+ return Error(404, notfound.Error(), nil)
}
- return ApiError(500, "Could not get markdown file", err)
+ return Error(500, "Could not get markdown file", err)
}
resp := Respond(200, content)
@@ -178,8 +178,8 @@ func ImportDashboard(c *m.ReqContext, apiCmd dtos.ImportDashboardCommand) Respon
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to import dashboard", err)
+ return Error(500, "Failed to import dashboard", err)
}
- return Json(200, cmd.Result)
+ return JSON(200, cmd.Result)
}
diff --git a/pkg/api/preferences.go b/pkg/api/preferences.go
index 2b85c3dcee1..26695130975 100644
--- a/pkg/api/preferences.go
+++ b/pkg/api/preferences.go
@@ -13,10 +13,10 @@ func SetHomeDashboard(c *m.ReqContext, cmd m.SavePreferencesCommand) Response {
cmd.OrgId = c.OrgId
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to set home dashboard", err)
+ return Error(500, "Failed to set home dashboard", err)
}
- return ApiSuccess("Home dashboard set")
+ return Success("Home dashboard set")
}
// GET /api/user/preferences
@@ -28,16 +28,16 @@ func getPreferencesFor(orgID int64, userID int64) Response {
prefsQuery := m.GetPreferencesQuery{UserId: userID, OrgId: orgID}
if err := bus.Dispatch(&prefsQuery); err != nil {
- return ApiError(500, "Failed to get preferences", err)
+ return Error(500, "Failed to get preferences", err)
}
dto := dtos.Prefs{
Theme: prefsQuery.Result.Theme,
- HomeDashboardId: prefsQuery.Result.HomeDashboardId,
+ HomeDashboardID: prefsQuery.Result.HomeDashboardId,
Timezone: prefsQuery.Result.Timezone,
}
- return Json(200, &dto)
+ return JSON(200, &dto)
}
// PUT /api/user/preferences
@@ -51,14 +51,14 @@ func updatePreferencesFor(orgID int64, userID int64, dtoCmd *dtos.UpdatePrefsCmd
OrgId: orgID,
Theme: dtoCmd.Theme,
Timezone: dtoCmd.Timezone,
- HomeDashboardId: dtoCmd.HomeDashboardId,
+ HomeDashboardId: dtoCmd.HomeDashboardID,
}
if err := bus.Dispatch(&saveCmd); err != nil {
- return ApiError(500, "Failed to save preferences", err)
+ return Error(500, "Failed to save preferences", err)
}
- return ApiSuccess("Preferences updated")
+ return Success("Preferences updated")
}
// GET /api/org/preferences
diff --git a/pkg/api/quota.go b/pkg/api/quota.go
index f92acaf470f..d469f843930 100644
--- a/pkg/api/quota.go
+++ b/pkg/api/quota.go
@@ -8,60 +8,60 @@ import (
func GetOrgQuotas(c *m.ReqContext) Response {
if !setting.Quota.Enabled {
- return ApiError(404, "Quotas not enabled", nil)
+ return Error(404, "Quotas not enabled", nil)
}
query := m.GetOrgQuotasQuery{OrgId: c.ParamsInt64(":orgId")}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Failed to get org quotas", err)
+ return Error(500, "Failed to get org quotas", err)
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
func UpdateOrgQuota(c *m.ReqContext, cmd m.UpdateOrgQuotaCmd) Response {
if !setting.Quota.Enabled {
- return ApiError(404, "Quotas not enabled", nil)
+ return Error(404, "Quotas not enabled", nil)
}
cmd.OrgId = c.ParamsInt64(":orgId")
cmd.Target = c.Params(":target")
if _, ok := setting.Quota.Org.ToMap()[cmd.Target]; !ok {
- return ApiError(404, "Invalid quota target", nil)
+ return Error(404, "Invalid quota target", nil)
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to update org quotas", err)
+ return Error(500, "Failed to update org quotas", err)
}
- return ApiSuccess("Organization quota updated")
+ return Success("Organization quota updated")
}
func GetUserQuotas(c *m.ReqContext) Response {
if !setting.Quota.Enabled {
- return ApiError(404, "Quotas not enabled", nil)
+ return Error(404, "Quotas not enabled", nil)
}
query := m.GetUserQuotasQuery{UserId: c.ParamsInt64(":id")}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Failed to get org quotas", err)
+ return Error(500, "Failed to get org quotas", err)
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
func UpdateUserQuota(c *m.ReqContext, cmd m.UpdateUserQuotaCmd) Response {
if !setting.Quota.Enabled {
- return ApiError(404, "Quotas not enabled", nil)
+ return Error(404, "Quotas not enabled", nil)
}
cmd.UserId = c.ParamsInt64(":id")
cmd.Target = c.Params(":target")
if _, ok := setting.Quota.User.ToMap()[cmd.Target]; !ok {
- return ApiError(404, "Invalid quota target", nil)
+ return Error(404, "Invalid quota target", nil)
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to update org quotas", err)
+ return Error(500, "Failed to update org quotas", err)
}
- return ApiSuccess("Organization quota updated")
+ return Success("Organization quota updated")
}
diff --git a/pkg/api/signup.go b/pkg/api/signup.go
index 838d2f9c0af..200a3ebc9d1 100644
--- a/pkg/api/signup.go
+++ b/pkg/api/signup.go
@@ -12,7 +12,7 @@ import (
// GET /api/user/signup/options
func GetSignUpOptions(c *m.ReqContext) Response {
- return Json(200, util.DynMap{
+ return JSON(200, util.DynMap{
"verifyEmailEnabled": setting.VerifyEmailEnabled,
"autoAssignOrg": setting.AutoAssignOrg,
})
@@ -21,12 +21,12 @@ func GetSignUpOptions(c *m.ReqContext) Response {
// POST /api/user/signup
func SignUp(c *m.ReqContext, form dtos.SignUpForm) Response {
if !setting.AllowUserSignUp {
- return ApiError(401, "User signup is disabled", nil)
+ return Error(401, "User signup is disabled", nil)
}
existing := m.GetUserByLoginQuery{LoginOrEmail: form.Email}
if err := bus.Dispatch(&existing); err == nil {
- return ApiError(422, "User with same email address already exists", nil)
+ return Error(422, "User with same email address already exists", nil)
}
cmd := m.CreateTempUserCommand{}
@@ -38,7 +38,7 @@ func SignUp(c *m.ReqContext, form dtos.SignUpForm) Response {
cmd.RemoteAddr = c.Req.RemoteAddr
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to create signup", err)
+ return Error(500, "Failed to create signup", err)
}
bus.Publish(&events.SignUpStarted{
@@ -48,12 +48,12 @@ func SignUp(c *m.ReqContext, form dtos.SignUpForm) Response {
metrics.M_Api_User_SignUpStarted.Inc()
- return Json(200, util.DynMap{"status": "SignUpCreated"})
+ return JSON(200, util.DynMap{"status": "SignUpCreated"})
}
func SignUpStep2(c *m.ReqContext, form dtos.SignUpStep2Form) Response {
if !setting.AllowUserSignUp {
- return ApiError(401, "User signup is disabled", nil)
+ return Error(401, "User signup is disabled", nil)
}
createUserCmd := m.CreateUserCommand{
@@ -75,12 +75,12 @@ func SignUpStep2(c *m.ReqContext, form dtos.SignUpStep2Form) Response {
// check if user exists
existing := m.GetUserByLoginQuery{LoginOrEmail: form.Email}
if err := bus.Dispatch(&existing); err == nil {
- return ApiError(401, "User with same email address already exists", nil)
+ return Error(401, "User with same email address already exists", nil)
}
// dispatch create command
if err := bus.Dispatch(&createUserCmd); err != nil {
- return ApiError(500, "Failed to create user", err)
+ return Error(500, "Failed to create user", err)
}
// publish signup event
@@ -98,7 +98,7 @@ func SignUpStep2(c *m.ReqContext, form dtos.SignUpStep2Form) Response {
// check for pending invites
invitesQuery := m.GetTempUsersQuery{Email: form.Email, Status: m.TmpUserInvitePending}
if err := bus.Dispatch(&invitesQuery); err != nil {
- return ApiError(500, "Failed to query database for invites", err)
+ return Error(500, "Failed to query database for invites", err)
}
apiResponse := util.DynMap{"message": "User sign up completed successfully", "code": "redirect-to-landing-page"}
@@ -112,7 +112,7 @@ func SignUpStep2(c *m.ReqContext, form dtos.SignUpStep2Form) Response {
loginUserWithUser(user, c)
metrics.M_Api_User_SignUpCompleted.Inc()
- return Json(200, apiResponse)
+ return JSON(200, apiResponse)
}
func verifyUserSignUpEmail(email string, code string) (bool, Response) {
@@ -120,14 +120,14 @@ func verifyUserSignUpEmail(email string, code string) (bool, Response) {
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrTempUserNotFound {
- return false, ApiError(404, "Invalid email verification code", nil)
+ return false, Error(404, "Invalid email verification code", nil)
}
- return false, ApiError(500, "Failed to read temp user", err)
+ return false, Error(500, "Failed to read temp user", err)
}
tempUser := query.Result
if tempUser.Email != email {
- return false, ApiError(404, "Email verification code does not match email", nil)
+ return false, Error(404, "Email verification code does not match email", nil)
}
return true, nil
diff --git a/pkg/api/stars.go b/pkg/api/stars.go
index 5361f64eea6..2c55b95dfbe 100644
--- a/pkg/api/stars.go
+++ b/pkg/api/stars.go
@@ -7,20 +7,20 @@ import (
func StarDashboard(c *m.ReqContext) Response {
if !c.IsSignedIn {
- return ApiError(412, "You need to sign in to star dashboards", nil)
+ return Error(412, "You need to sign in to star dashboards", nil)
}
cmd := m.StarDashboardCommand{UserId: c.UserId, DashboardId: c.ParamsInt64(":id")}
if cmd.DashboardId <= 0 {
- return ApiError(400, "Missing dashboard id", nil)
+ return Error(400, "Missing dashboard id", nil)
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to star dashboard", err)
+ return Error(500, "Failed to star dashboard", err)
}
- return ApiSuccess("Dashboard starred!")
+ return Success("Dashboard starred!")
}
func UnstarDashboard(c *m.ReqContext) Response {
@@ -28,12 +28,12 @@ func UnstarDashboard(c *m.ReqContext) Response {
cmd := m.UnstarDashboardCommand{UserId: c.UserId, DashboardId: c.ParamsInt64(":id")}
if cmd.DashboardId <= 0 {
- return ApiError(400, "Missing dashboard id", nil)
+ return Error(400, "Missing dashboard id", nil)
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to unstar dashboard", err)
+ return Error(500, "Failed to unstar dashboard", err)
}
- return ApiSuccess("Dashboard unstarred")
+ return Success("Dashboard unstarred")
}
diff --git a/pkg/api/team.go b/pkg/api/team.go
index 13e23df27c8..9919305881b 100644
--- a/pkg/api/team.go
+++ b/pkg/api/team.go
@@ -12,12 +12,12 @@ func CreateTeam(c *m.ReqContext, cmd m.CreateTeamCommand) Response {
cmd.OrgId = c.OrgId
if err := bus.Dispatch(&cmd); err != nil {
if err == m.ErrTeamNameTaken {
- return ApiError(409, "Team name taken", err)
+ return Error(409, "Team name taken", err)
}
- return ApiError(500, "Failed to create Team", err)
+ return Error(500, "Failed to create Team", err)
}
- return Json(200, &util.DynMap{
+ return JSON(200, &util.DynMap{
"teamId": cmd.Result.Id,
"message": "Team created",
})
@@ -29,23 +29,23 @@ func UpdateTeam(c *m.ReqContext, cmd m.UpdateTeamCommand) Response {
cmd.Id = c.ParamsInt64(":teamId")
if err := bus.Dispatch(&cmd); err != nil {
if err == m.ErrTeamNameTaken {
- return ApiError(400, "Team name taken", err)
+ return Error(400, "Team name taken", err)
}
- return ApiError(500, "Failed to update Team", err)
+ return Error(500, "Failed to update Team", err)
}
- return ApiSuccess("Team updated")
+ return Success("Team updated")
}
// DELETE /api/teams/:teamId
func DeleteTeamByID(c *m.ReqContext) Response {
if err := bus.Dispatch(&m.DeleteTeamCommand{OrgId: c.OrgId, Id: c.ParamsInt64(":teamId")}); err != nil {
if err == m.ErrTeamNotFound {
- return ApiError(404, "Failed to delete Team. ID not found", nil)
+ return Error(404, "Failed to delete Team. ID not found", nil)
}
- return ApiError(500, "Failed to update Team", err)
+ return Error(500, "Failed to update Team", err)
}
- return ApiSuccess("Team deleted")
+ return Success("Team deleted")
}
// GET /api/teams/search
@@ -68,7 +68,7 @@ func SearchTeams(c *m.ReqContext) Response {
}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Failed to search Teams", err)
+ return Error(500, "Failed to search Teams", err)
}
for _, team := range query.Result.Teams {
@@ -78,7 +78,7 @@ func SearchTeams(c *m.ReqContext) Response {
query.Result.Page = page
query.Result.PerPage = perPage
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
// GET /api/teams/:teamId
@@ -87,11 +87,11 @@ func GetTeamByID(c *m.ReqContext) Response {
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrTeamNotFound {
- return ApiError(404, "Team not found", err)
+ return Error(404, "Team not found", err)
}
- return ApiError(500, "Failed to get Team", err)
+ return Error(500, "Failed to get Team", err)
}
- return Json(200, &query.Result)
+ return JSON(200, &query.Result)
}
diff --git a/pkg/api/team_members.go b/pkg/api/team_members.go
index 4fb05b016e3..60a170a8c31 100644
--- a/pkg/api/team_members.go
+++ b/pkg/api/team_members.go
@@ -12,14 +12,14 @@ func GetTeamMembers(c *m.ReqContext) Response {
query := m.GetTeamMembersQuery{OrgId: c.OrgId, TeamId: c.ParamsInt64(":teamId")}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Failed to get Team Members", err)
+ return Error(500, "Failed to get Team Members", err)
}
for _, member := range query.Result {
member.AvatarUrl = dtos.GetGravatarUrl(member.Email)
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
// POST /api/teams/:teamId/members
@@ -29,17 +29,17 @@ func AddTeamMember(c *m.ReqContext, cmd m.AddTeamMemberCommand) Response {
if err := bus.Dispatch(&cmd); err != nil {
if err == m.ErrTeamNotFound {
- return ApiError(404, "Team not found", nil)
+ return Error(404, "Team not found", nil)
}
if err == m.ErrTeamMemberAlreadyAdded {
- return ApiError(400, "User is already added to this team", nil)
+ return Error(400, "User is already added to this team", nil)
}
- return ApiError(500, "Failed to add Member to Team", err)
+ return Error(500, "Failed to add Member to Team", err)
}
- return Json(200, &util.DynMap{
+ return JSON(200, &util.DynMap{
"message": "Member added to Team",
})
}
@@ -48,14 +48,14 @@ func AddTeamMember(c *m.ReqContext, cmd m.AddTeamMemberCommand) Response {
func RemoveTeamMember(c *m.ReqContext) Response {
if err := bus.Dispatch(&m.RemoveTeamMemberCommand{OrgId: c.OrgId, TeamId: c.ParamsInt64(":teamId"), UserId: c.ParamsInt64(":userId")}); err != nil {
if err == m.ErrTeamNotFound {
- return ApiError(404, "Team not found", nil)
+ return Error(404, "Team not found", nil)
}
if err == m.ErrTeamMemberNotFound {
- return ApiError(404, "Team member not found", nil)
+ return Error(404, "Team member not found", nil)
}
- return ApiError(500, "Failed to remove Member from Team", err)
+ return Error(500, "Failed to remove Member from Team", err)
}
- return ApiSuccess("Team Member removed")
+ return Success("Team Member removed")
}
diff --git a/pkg/api/user.go b/pkg/api/user.go
index 4469fab2d29..725c623575f 100644
--- a/pkg/api/user.go
+++ b/pkg/api/user.go
@@ -23,12 +23,12 @@ func getUserUserProfile(userID int64) Response {
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrUserNotFound {
- return ApiError(404, m.ErrUserNotFound.Error(), nil)
+ return Error(404, m.ErrUserNotFound.Error(), nil)
}
- return ApiError(500, "Failed to get user", err)
+ return Error(500, "Failed to get user", err)
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
// GET /api/users/lookup
@@ -36,9 +36,9 @@ func GetUserByLoginOrEmail(c *m.ReqContext) Response {
query := m.GetUserByLoginQuery{LoginOrEmail: c.Query("loginOrEmail")}
if err := bus.Dispatch(&query); err != nil {
if err == m.ErrUserNotFound {
- return ApiError(404, m.ErrUserNotFound.Error(), nil)
+ return Error(404, m.ErrUserNotFound.Error(), nil)
}
- return ApiError(500, "Failed to get user", err)
+ return Error(500, "Failed to get user", err)
}
user := query.Result
result := m.UserProfileDTO{
@@ -50,17 +50,17 @@ func GetUserByLoginOrEmail(c *m.ReqContext) Response {
IsGrafanaAdmin: user.IsAdmin,
OrgId: user.OrgId,
}
- return Json(200, &result)
+ return JSON(200, &result)
}
// POST /api/user
func UpdateSignedInUser(c *m.ReqContext, cmd m.UpdateUserCommand) Response {
if setting.AuthProxyEnabled {
if setting.AuthProxyHeaderProperty == "email" && cmd.Email != c.Email {
- return ApiError(400, "Not allowed to change email when auth proxy is using email property", nil)
+ return Error(400, "Not allowed to change email when auth proxy is using email property", nil)
}
if setting.AuthProxyHeaderProperty == "username" && cmd.Login != c.Login {
- return ApiError(400, "Not allowed to change username when auth proxy is using username property", nil)
+ return Error(400, "Not allowed to change username when auth proxy is using username property", nil)
}
}
cmd.UserId = c.UserId
@@ -79,31 +79,31 @@ func UpdateUserActiveOrg(c *m.ReqContext) Response {
orgID := c.ParamsInt64(":orgId")
if !validateUsingOrg(userID, orgID) {
- return ApiError(401, "Not a valid organization", nil)
+ return Error(401, "Not a valid organization", nil)
}
cmd := m.SetUsingOrgCommand{UserId: userID, OrgId: orgID}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to change active organization", err)
+ return Error(500, "Failed to change active organization", err)
}
- return ApiSuccess("Active organization changed")
+ return Success("Active organization changed")
}
func handleUpdateUser(cmd m.UpdateUserCommand) Response {
if len(cmd.Login) == 0 {
cmd.Login = cmd.Email
if len(cmd.Login) == 0 {
- return ApiError(400, "Validation error, need to specify either username or email", nil)
+ return Error(400, "Validation error, need to specify either username or email", nil)
}
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to update user", err)
+ return Error(500, "Failed to update user", err)
}
- return ApiSuccess("User updated")
+ return Success("User updated")
}
// GET /api/user/orgs
@@ -120,10 +120,10 @@ func getUserOrgList(userID int64) Response {
query := m.GetUserOrgListQuery{UserId: userID}
if err := bus.Dispatch(&query); err != nil {
- return ApiError(500, "Failed to get user organizations", err)
+ return Error(500, "Failed to get user organizations", err)
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
func validateUsingOrg(userID int64, orgID int64) bool {
@@ -149,16 +149,16 @@ func UserSetUsingOrg(c *m.ReqContext) Response {
orgID := c.ParamsInt64(":id")
if !validateUsingOrg(c.UserId, orgID) {
- return ApiError(401, "Not a valid organization", nil)
+ return Error(401, "Not a valid organization", nil)
}
cmd := m.SetUsingOrgCommand{UserId: c.UserId, OrgId: orgID}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to change active organization", err)
+ return Error(500, "Failed to change active organization", err)
}
- return ApiSuccess("Active organization changed")
+ return Success("Active organization changed")
}
// GET /profile/switch-org/:id
@@ -180,53 +180,53 @@ func ChangeActiveOrgAndRedirectToHome(c *m.ReqContext) {
func ChangeUserPassword(c *m.ReqContext, cmd m.ChangeUserPasswordCommand) Response {
if setting.LdapEnabled || setting.AuthProxyEnabled {
- return ApiError(400, "Not allowed to change password when LDAP or Auth Proxy is enabled", nil)
+ return Error(400, "Not allowed to change password when LDAP or Auth Proxy is enabled", nil)
}
userQuery := m.GetUserByIdQuery{Id: c.UserId}
if err := bus.Dispatch(&userQuery); err != nil {
- return ApiError(500, "Could not read user from database", err)
+ return Error(500, "Could not read user from database", err)
}
passwordHashed := util.EncodePassword(cmd.OldPassword, userQuery.Result.Salt)
if passwordHashed != userQuery.Result.Password {
- return ApiError(401, "Invalid old password", nil)
+ return Error(401, "Invalid old password", nil)
}
password := m.Password(cmd.NewPassword)
if password.IsWeak() {
- return ApiError(400, "New password is too short", nil)
+ return Error(400, "New password is too short", nil)
}
cmd.UserId = c.UserId
cmd.NewPassword = util.EncodePassword(cmd.NewPassword, userQuery.Result.Salt)
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to change user password", err)
+ return Error(500, "Failed to change user password", err)
}
- return ApiSuccess("User password changed")
+ return Success("User password changed")
}
// GET /api/users
func SearchUsers(c *m.ReqContext) Response {
query, err := searchUser(c)
if err != nil {
- return ApiError(500, "Failed to fetch users", err)
+ return Error(500, "Failed to fetch users", err)
}
- return Json(200, query.Result.Users)
+ return JSON(200, query.Result.Users)
}
// GET /api/users/search
func SearchUsersWithPaging(c *m.ReqContext) Response {
query, err := searchUser(c)
if err != nil {
- return ApiError(500, "Failed to fetch users", err)
+ return Error(500, "Failed to fetch users", err)
}
- return Json(200, query.Result)
+ return JSON(200, query.Result)
}
func searchUser(c *m.ReqContext) (*m.SearchUsersQuery, error) {
@@ -269,10 +269,10 @@ func SetHelpFlag(c *m.ReqContext) Response {
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to update help flag", err)
+ return Error(500, "Failed to update help flag", err)
}
- return Json(200, &util.DynMap{"message": "Help flag set", "helpFlags1": cmd.HelpFlags1})
+ return JSON(200, &util.DynMap{"message": "Help flag set", "helpFlags1": cmd.HelpFlags1})
}
func ClearHelpFlags(c *m.ReqContext) Response {
@@ -282,8 +282,8 @@ func ClearHelpFlags(c *m.ReqContext) Response {
}
if err := bus.Dispatch(&cmd); err != nil {
- return ApiError(500, "Failed to update help flag", err)
+ return Error(500, "Failed to update help flag", err)
}
- return Json(200, &util.DynMap{"message": "Help flag set", "helpFlags1": cmd.HelpFlags1})
+ return JSON(200, &util.DynMap{"message": "Help flag set", "helpFlags1": cmd.HelpFlags1})
}
diff --git a/pkg/cmd/grafana-server/server.go b/pkg/cmd/grafana-server/server.go
index 0338b3d6aa2..5bbf43087ec 100644
--- a/pkg/cmd/grafana-server/server.go
+++ b/pkg/cmd/grafana-server/server.go
@@ -49,7 +49,7 @@ type GrafanaServerImpl struct {
childRoutines *errgroup.Group
log log.Logger
- httpServer *api.HttpServer
+ httpServer *api.HTTPServer
}
func (g *GrafanaServerImpl) Start() error {
diff --git a/pkg/middleware/auth_proxy.go b/pkg/middleware/auth_proxy.go
index 4d2a7a98908..e1801404453 100644
--- a/pkg/middleware/auth_proxy.go
+++ b/pkg/middleware/auth_proxy.go
@@ -14,7 +14,7 @@ import (
"github.com/grafana/grafana/pkg/setting"
)
-func initContextWithAuthProxy(ctx *m.ReqContext, orgId int64) bool {
+func initContextWithAuthProxy(ctx *m.ReqContext, orgID int64) bool {
if !setting.AuthProxyEnabled {
return false
}
@@ -31,31 +31,31 @@ func initContextWithAuthProxy(ctx *m.ReqContext, orgId int64) bool {
}
query := getSignedInUserQueryForProxyAuth(proxyHeaderValue)
- query.OrgId = orgId
+ query.OrgId = orgID
if err := bus.Dispatch(query); err != nil {
if err != m.ErrUserNotFound {
ctx.Handle(500, "Failed to find user specified in auth proxy header", err)
return true
}
- if setting.AuthProxyAutoSignUp {
- cmd := getCreateUserCommandForProxyAuth(proxyHeaderValue)
- if setting.LdapEnabled {
- cmd.SkipOrgSetup = true
- }
-
- if err := bus.Dispatch(cmd); err != nil {
- ctx.Handle(500, "Failed to create user specified in auth proxy header", err)
- return true
- }
- query = &m.GetSignedInUserQuery{UserId: cmd.Result.Id, OrgId: orgId}
- if err := bus.Dispatch(query); err != nil {
- ctx.Handle(500, "Failed find user after creation", err)
- return true
- }
- } else {
+ if !setting.AuthProxyAutoSignUp {
return false
}
+
+ cmd := getCreateUserCommandForProxyAuth(proxyHeaderValue)
+ if setting.LdapEnabled {
+ cmd.SkipOrgSetup = true
+ }
+
+ if err := bus.Dispatch(cmd); err != nil {
+ ctx.Handle(500, "Failed to create user specified in auth proxy header", err)
+ return true
+ }
+ query = &m.GetSignedInUserQuery{UserId: cmd.Result.Id, OrgId: orgID}
+ if err := bus.Dispatch(query); err != nil {
+ ctx.Handle(500, "Failed find user after creation", err)
+ return true
+ }
}
// initialize session
@@ -96,50 +96,53 @@ func initContextWithAuthProxy(ctx *m.ReqContext, orgId int64) bool {
}
var syncGrafanaUserWithLdapUser = func(ctx *m.ReqContext, query *m.GetSignedInUserQuery) error {
- if setting.LdapEnabled {
- expireEpoch := time.Now().Add(time.Duration(-setting.AuthProxyLdapSyncTtl) * time.Minute).Unix()
+ if !setting.LdapEnabled {
+ return nil
+ }
- var lastLdapSync int64
- if lastLdapSyncInSession := ctx.Session.Get(session.SESS_KEY_LASTLDAPSYNC); lastLdapSyncInSession != nil {
- lastLdapSync = lastLdapSyncInSession.(int64)
- }
+ expireEpoch := time.Now().Add(time.Duration(-setting.AuthProxyLdapSyncTtl) * time.Minute).Unix()
- if lastLdapSync < expireEpoch {
- ldapCfg := login.LdapCfg
+ var lastLdapSync int64
+ if lastLdapSyncInSession := ctx.Session.Get(session.SESS_KEY_LASTLDAPSYNC); lastLdapSyncInSession != nil {
+ lastLdapSync = lastLdapSyncInSession.(int64)
+ }
- for _, server := range ldapCfg.Servers {
- author := login.NewLdapAuthenticator(server)
- if err := author.SyncSignedInUser(query.Result); err != nil {
- return err
- }
+ if lastLdapSync < expireEpoch {
+ ldapCfg := login.LdapCfg
+
+ for _, server := range ldapCfg.Servers {
+ author := login.NewLdapAuthenticator(server)
+ if err := author.SyncSignedInUser(query.Result); err != nil {
+ return err
}
-
- ctx.Session.Set(session.SESS_KEY_LASTLDAPSYNC, time.Now().Unix())
}
+
+ ctx.Session.Set(session.SESS_KEY_LASTLDAPSYNC, time.Now().Unix())
}
return nil
}
func checkAuthenticationProxy(ctx *m.ReqContext, proxyHeaderValue string) error {
- if len(strings.TrimSpace(setting.AuthProxyWhitelist)) > 0 {
- proxies := strings.Split(setting.AuthProxyWhitelist, ",")
- remoteAddrSplit := strings.Split(ctx.Req.RemoteAddr, ":")
- sourceIP := remoteAddrSplit[0]
+ if len(strings.TrimSpace(setting.AuthProxyWhitelist)) == 0 {
+ return nil
+ }
+ proxies := strings.Split(setting.AuthProxyWhitelist, ",")
+ remoteAddrSplit := strings.Split(ctx.Req.RemoteAddr, ":")
+ sourceIP := remoteAddrSplit[0]
- found := false
- for _, proxyIP := range proxies {
- if sourceIP == strings.TrimSpace(proxyIP) {
- found = true
- break
- }
+ found := false
+ for _, proxyIP := range proxies {
+ if sourceIP == strings.TrimSpace(proxyIP) {
+ found = true
+ break
}
+ }
- if !found {
- msg := fmt.Sprintf("Request for user (%s) is not from the authentication proxy", proxyHeaderValue)
- err := errors.New(msg)
- return err
- }
+ if !found {
+ msg := fmt.Sprintf("Request for user (%s) is not from the authentication proxy", proxyHeaderValue)
+ err := errors.New(msg)
+ return err
}
return nil
diff --git a/pkg/middleware/dashboard_redirect.go b/pkg/middleware/dashboard_redirect.go
index 024b112e154..2edf04d543e 100644
--- a/pkg/middleware/dashboard_redirect.go
+++ b/pkg/middleware/dashboard_redirect.go
@@ -10,8 +10,8 @@ import (
"gopkg.in/macaron.v1"
)
-func getDashboardUrlBySlug(orgId int64, slug string) (string, error) {
- query := m.GetDashboardQuery{Slug: slug, OrgId: orgId}
+func getDashboardURLBySlug(orgID int64, slug string) (string, error) {
+ query := m.GetDashboardQuery{Slug: slug, OrgId: orgID}
if err := bus.Dispatch(&query); err != nil {
return "", m.ErrDashboardNotFound
@@ -25,7 +25,7 @@ func RedirectFromLegacyDashboardURL() macaron.Handler {
slug := c.Params("slug")
if slug != "" {
- if url, err := getDashboardUrlBySlug(c.OrgId, slug); err == nil {
+ if url, err := getDashboardURLBySlug(c.OrgId, slug); err == nil {
url = fmt.Sprintf("%s?%s", url, c.Req.URL.RawQuery)
c.Redirect(url, 301)
return
@@ -34,13 +34,13 @@ func RedirectFromLegacyDashboardURL() macaron.Handler {
}
}
-func RedirectFromLegacyDashboardSoloUrl() macaron.Handler {
+func RedirectFromLegacyDashboardSoloURL() macaron.Handler {
return func(c *m.ReqContext) {
slug := c.Params("slug")
renderRequest := c.QueryBool("render")
if slug != "" {
- if url, err := getDashboardUrlBySlug(c.OrgId, slug); err == nil {
+ if url, err := getDashboardURLBySlug(c.OrgId, slug); err == nil {
if renderRequest && strings.Contains(url, setting.AppSubUrl) {
url = strings.Replace(url, setting.AppSubUrl, "", 1)
}
diff --git a/pkg/middleware/dashboard_redirect_test.go b/pkg/middleware/dashboard_redirect_test.go
index 24eab2d7b79..1e4d8293cae 100644
--- a/pkg/middleware/dashboard_redirect_test.go
+++ b/pkg/middleware/dashboard_redirect_test.go
@@ -14,7 +14,7 @@ func TestMiddlewareDashboardRedirect(t *testing.T) {
Convey("Given the dashboard redirect middleware", t, func() {
bus.ClearBusHandlers()
redirectFromLegacyDashboardUrl := RedirectFromLegacyDashboardURL()
- redirectFromLegacyDashboardSoloUrl := RedirectFromLegacyDashboardSoloUrl()
+ redirectFromLegacyDashboardSoloUrl := RedirectFromLegacyDashboardSoloURL()
fakeDash := m.NewDashboard("Child dash")
fakeDash.Id = 1
diff --git a/pkg/middleware/render_auth.go b/pkg/middleware/render_auth.go
index 225645e659e..6c338becbda 100644
--- a/pkg/middleware/render_auth.go
+++ b/pkg/middleware/render_auth.go
@@ -19,16 +19,16 @@ func initContextWithRenderAuth(ctx *m.ReqContext) bool {
renderKeysLock.Lock()
defer renderKeysLock.Unlock()
- if renderUser, exists := renderKeys[key]; !exists {
+ renderUser, exists := renderKeys[key]
+ if !exists {
ctx.JsonApiErr(401, "Invalid Render Key", nil)
return true
- } else {
-
- ctx.IsSignedIn = true
- ctx.SignedInUser = renderUser
- ctx.IsRenderCall = true
- return true
}
+
+ ctx.IsSignedIn = true
+ ctx.SignedInUser = renderUser
+ ctx.IsRenderCall = true
+ return true
}
type renderContextFunc func(key string) (string, error)
diff --git a/pkg/services/alerting/engine.go b/pkg/services/alerting/engine.go
index 4448a5cb978..a6f97333d1b 100644
--- a/pkg/services/alerting/engine.go
+++ b/pkg/services/alerting/engine.go
@@ -86,17 +86,63 @@ func (e *Engine) runJobDispatcher(grafanaCtx context.Context) error {
case <-grafanaCtx.Done():
return dispatcherGroup.Wait()
case job := <-e.execQueue:
- dispatcherGroup.Go(func() error { return e.processJob(alertCtx, job) })
+ dispatcherGroup.Go(func() error { return e.processJobWithRetry(alertCtx, job) })
}
}
}
var (
unfinishedWorkTimeout time.Duration = time.Second * 5
- alertTimeout time.Duration = time.Second * 30
+ // TODO: Make alertTimeout and alertMaxAttempts configurable in the config file.
+ alertTimeout time.Duration = time.Second * 30
+ alertMaxAttempts int = 3
)
-func (e *Engine) processJob(grafanaCtx context.Context, job *Job) error {
+func (e *Engine) processJobWithRetry(grafanaCtx context.Context, job *Job) error {
+ defer func() {
+ if err := recover(); err != nil {
+ e.log.Error("Alert Panic", "error", err, "stack", log.Stack(1))
+ }
+ }()
+
+ cancelChan := make(chan context.CancelFunc, alertMaxAttempts)
+ attemptChan := make(chan int, 1)
+
+ // Initialize with first attemptID=1
+ attemptChan <- 1
+ job.Running = true
+
+ for {
+ select {
+ case <-grafanaCtx.Done():
+ // In case grafana server context is cancel, let a chance to job processing
+ // to finish gracefully - by waiting a timeout duration - before forcing its end.
+ unfinishedWorkTimer := time.NewTimer(unfinishedWorkTimeout)
+ select {
+ case <-unfinishedWorkTimer.C:
+ return e.endJob(grafanaCtx.Err(), cancelChan, job)
+ case <-attemptChan:
+ return e.endJob(nil, cancelChan, job)
+ }
+ case attemptID, more := <-attemptChan:
+ if !more {
+ return e.endJob(nil, cancelChan, job)
+ }
+ go e.processJob(attemptID, attemptChan, cancelChan, job)
+ }
+ }
+}
+
+func (e *Engine) endJob(err error, cancelChan chan context.CancelFunc, job *Job) error {
+ job.Running = false
+ close(cancelChan)
+ for cancelFn := range cancelChan {
+ cancelFn()
+ }
+ return err
+}
+
+func (e *Engine) processJob(attemptID int, attemptChan chan int, cancelChan chan context.CancelFunc, job *Job) {
defer func() {
if err := recover(); err != nil {
e.log.Error("Alert Panic", "error", err, "stack", log.Stack(1))
@@ -104,14 +150,13 @@ func (e *Engine) processJob(grafanaCtx context.Context, job *Job) error {
}()
alertCtx, cancelFn := context.WithTimeout(context.Background(), alertTimeout)
+ cancelChan <- cancelFn
span := opentracing.StartSpan("alert execution")
alertCtx = opentracing.ContextWithSpan(alertCtx, span)
- job.Running = true
evalContext := NewEvalContext(alertCtx, job.Rule)
evalContext.Ctx = alertCtx
- done := make(chan struct{})
go func() {
defer func() {
if err := recover(); err != nil {
@@ -122,43 +167,36 @@ func (e *Engine) processJob(grafanaCtx context.Context, job *Job) error {
tlog.String("message", "failed to execute alert rule. panic was recovered."),
)
span.Finish()
- close(done)
+ close(attemptChan)
}
}()
e.evalHandler.Eval(evalContext)
- e.resultHandler.Handle(evalContext)
span.SetTag("alertId", evalContext.Rule.Id)
span.SetTag("dashboardId", evalContext.Rule.DashboardId)
span.SetTag("firing", evalContext.Firing)
span.SetTag("nodatapoints", evalContext.NoDataFound)
+ span.SetTag("attemptID", attemptID)
+
if evalContext.Error != nil {
ext.Error.Set(span, true)
span.LogFields(
tlog.Error(evalContext.Error),
- tlog.String("message", "alerting execution failed"),
+ tlog.String("message", "alerting execution attempt failed"),
)
+ if attemptID < alertMaxAttempts {
+ span.Finish()
+ e.log.Debug("Job Execution attempt triggered retry", "timeMs", evalContext.GetDurationMs(), "alertId", evalContext.Rule.Id, "name", evalContext.Rule.Name, "firing", evalContext.Firing, "attemptID", attemptID)
+ attemptChan <- (attemptID + 1)
+ return
+ }
}
+ evalContext.Rule.State = evalContext.GetNewState()
+ e.resultHandler.Handle(evalContext)
span.Finish()
- close(done)
+ e.log.Debug("Job Execution completed", "timeMs", evalContext.GetDurationMs(), "alertId", evalContext.Rule.Id, "name", evalContext.Rule.Name, "firing", evalContext.Firing, "attemptID", attemptID)
+ close(attemptChan)
}()
-
- var err error = nil
- select {
- case <-grafanaCtx.Done():
- select {
- case <-time.After(unfinishedWorkTimeout):
- cancelFn()
- err = grafanaCtx.Err()
- case <-done:
- }
- case <-done:
- }
-
- e.log.Debug("Job Execution completed", "timeMs", evalContext.GetDurationMs(), "alertId", evalContext.Rule.Id, "name", evalContext.Rule.Name, "firing", evalContext.Firing)
- job.Running = false
- cancelFn()
- return err
}
diff --git a/pkg/services/alerting/engine_test.go b/pkg/services/alerting/engine_test.go
new file mode 100644
index 00000000000..64f954c6dd5
--- /dev/null
+++ b/pkg/services/alerting/engine_test.go
@@ -0,0 +1,118 @@
+package alerting
+
+import (
+ "context"
+ "errors"
+ "math"
+ "testing"
+
+ . "github.com/smartystreets/goconvey/convey"
+)
+
+type FakeEvalHandler struct {
+ SuccessCallID int // 0 means never sucess
+ CallNb int
+}
+
+func NewFakeEvalHandler(successCallID int) *FakeEvalHandler {
+ return &FakeEvalHandler{
+ SuccessCallID: successCallID,
+ CallNb: 0,
+ }
+}
+
+func (handler *FakeEvalHandler) Eval(evalContext *EvalContext) {
+ handler.CallNb++
+ if handler.CallNb != handler.SuccessCallID {
+ evalContext.Error = errors.New("Fake evaluation failure")
+ }
+}
+
+type FakeResultHandler struct{}
+
+func (handler *FakeResultHandler) Handle(evalContext *EvalContext) error {
+ return nil
+}
+
+func TestEngineProcessJob(t *testing.T) {
+ Convey("Alerting engine job processing", t, func() {
+ engine := NewEngine()
+ engine.resultHandler = &FakeResultHandler{}
+ job := &Job{Running: true, Rule: &Rule{}}
+
+ Convey("Should trigger retry if needed", func() {
+
+ Convey("error + not last attempt -> retry", func() {
+ engine.evalHandler = NewFakeEvalHandler(0)
+
+ for i := 1; i < alertMaxAttempts; i++ {
+ attemptChan := make(chan int, 1)
+ cancelChan := make(chan context.CancelFunc, alertMaxAttempts)
+
+ engine.processJob(i, attemptChan, cancelChan, job)
+ nextAttemptID, more := <-attemptChan
+
+ So(nextAttemptID, ShouldEqual, i+1)
+ So(more, ShouldEqual, true)
+ So(<-cancelChan, ShouldNotBeNil)
+ }
+ })
+
+ Convey("error + last attempt -> no retry", func() {
+ engine.evalHandler = NewFakeEvalHandler(0)
+ attemptChan := make(chan int, 1)
+ cancelChan := make(chan context.CancelFunc, alertMaxAttempts)
+
+ engine.processJob(alertMaxAttempts, attemptChan, cancelChan, job)
+ nextAttemptID, more := <-attemptChan
+
+ So(nextAttemptID, ShouldEqual, 0)
+ So(more, ShouldEqual, false)
+ So(<-cancelChan, ShouldNotBeNil)
+ })
+
+ Convey("no error -> no retry", func() {
+ engine.evalHandler = NewFakeEvalHandler(1)
+ attemptChan := make(chan int, 1)
+ cancelChan := make(chan context.CancelFunc, alertMaxAttempts)
+
+ engine.processJob(1, attemptChan, cancelChan, job)
+ nextAttemptID, more := <-attemptChan
+
+ So(nextAttemptID, ShouldEqual, 0)
+ So(more, ShouldEqual, false)
+ So(<-cancelChan, ShouldNotBeNil)
+ })
+ })
+
+ Convey("Should trigger as many retries as needed", func() {
+
+ Convey("never sucess -> max retries number", func() {
+ expectedAttempts := alertMaxAttempts
+ evalHandler := NewFakeEvalHandler(0)
+ engine.evalHandler = evalHandler
+
+ engine.processJobWithRetry(context.TODO(), job)
+ So(evalHandler.CallNb, ShouldEqual, expectedAttempts)
+ })
+
+ Convey("always sucess -> never retry", func() {
+ expectedAttempts := 1
+ evalHandler := NewFakeEvalHandler(1)
+ engine.evalHandler = evalHandler
+
+ engine.processJobWithRetry(context.TODO(), job)
+ So(evalHandler.CallNb, ShouldEqual, expectedAttempts)
+ })
+
+ Convey("some errors before sucess -> some retries", func() {
+ expectedAttempts := int(math.Ceil(float64(alertMaxAttempts) / 2))
+ evalHandler := NewFakeEvalHandler(expectedAttempts)
+ engine.evalHandler = evalHandler
+
+ engine.processJobWithRetry(context.TODO(), job)
+ So(evalHandler.CallNb, ShouldEqual, expectedAttempts)
+ })
+ })
+ })
+}
diff --git a/pkg/services/alerting/eval_context.go b/pkg/services/alerting/eval_context.go
index d598203d675..91d0e179a14 100644
--- a/pkg/services/alerting/eval_context.go
+++ b/pkg/services/alerting/eval_context.go
@@ -112,3 +112,34 @@ func (c *EvalContext) GetRuleUrl() (string, error) {
return fmt.Sprintf(urlFormat, m.GetFullDashboardUrl(ref.Uid, ref.Slug), c.Rule.PanelId, c.Rule.OrgId), nil
}
}
+
+func (c *EvalContext) GetNewState() m.AlertStateType {
+ if c.Error != nil {
+ c.log.Error("Alert Rule Result Error",
+ "ruleId", c.Rule.Id,
+ "name", c.Rule.Name,
+ "error", c.Error,
+ "changing state to", c.Rule.ExecutionErrorState.ToAlertState())
+
+ if c.Rule.ExecutionErrorState == m.ExecutionErrorKeepState {
+ return c.PrevAlertState
+ }
+ return c.Rule.ExecutionErrorState.ToAlertState()
+
+ } else if c.Firing {
+ return m.AlertStateAlerting
+
+ } else if c.NoDataFound {
+ c.log.Info("Alert Rule returned no data",
+ "ruleId", c.Rule.Id,
+ "name", c.Rule.Name,
+ "changing state to", c.Rule.NoDataState.ToAlertState())
+
+ if c.Rule.NoDataState == m.NoDataKeepState {
+ return c.PrevAlertState
+ }
+ return c.Rule.NoDataState.ToAlertState()
+ }
+
+ return m.AlertStateOK
+}
diff --git a/pkg/services/alerting/eval_context_test.go b/pkg/services/alerting/eval_context_test.go
index 019ca1ed01f..709eeee4e5e 100644
--- a/pkg/services/alerting/eval_context_test.go
+++ b/pkg/services/alerting/eval_context_test.go
@@ -2,6 +2,7 @@ package alerting
import (
"context"
+ "fmt"
"testing"
"github.com/grafana/grafana/pkg/models"
@@ -12,7 +13,7 @@ func TestAlertingEvalContext(t *testing.T) {
Convey("Eval context", t, func() {
ctx := NewEvalContext(context.TODO(), &Rule{Conditions: []Condition{&conditionStub{firing: true}}})
- Convey("Should update alert state", func() {
+ Convey("Should update alert state when needed", func() {
Convey("ok -> alerting", func() {
ctx.PrevAlertState = models.AlertStateOK
@@ -28,5 +29,71 @@ func TestAlertingEvalContext(t *testing.T) {
So(ctx.ShouldUpdateAlertState(), ShouldBeFalse)
})
})
+
+ Convey("Should compute and replace properly new rule state", func() {
+ dummieError := fmt.Errorf("dummie error")
+
+ Convey("ok -> alerting", func() {
+ ctx.PrevAlertState = models.AlertStateOK
+ ctx.Firing = true
+
+ ctx.Rule.State = ctx.GetNewState()
+ So(ctx.Rule.State, ShouldEqual, models.AlertStateAlerting)
+ })
+
+ Convey("ok -> error(alerting)", func() {
+ ctx.PrevAlertState = models.AlertStateOK
+ ctx.Error = dummieError
+ ctx.Rule.ExecutionErrorState = models.ExecutionErrorSetAlerting
+
+ ctx.Rule.State = ctx.GetNewState()
+ So(ctx.Rule.State, ShouldEqual, models.AlertStateAlerting)
+ })
+
+ Convey("ok -> error(keep_last)", func() {
+ ctx.PrevAlertState = models.AlertStateOK
+ ctx.Error = dummieError
+ ctx.Rule.ExecutionErrorState = models.ExecutionErrorKeepState
+
+ ctx.Rule.State = ctx.GetNewState()
+ So(ctx.Rule.State, ShouldEqual, models.AlertStateOK)
+ })
+
+ Convey("pending -> error(keep_last)", func() {
+ ctx.PrevAlertState = models.AlertStatePending
+ ctx.Error = dummieError
+ ctx.Rule.ExecutionErrorState = models.ExecutionErrorKeepState
+
+ ctx.Rule.State = ctx.GetNewState()
+ So(ctx.Rule.State, ShouldEqual, models.AlertStatePending)
+ })
+
+ Convey("ok -> no_data(alerting)", func() {
+ ctx.PrevAlertState = models.AlertStateOK
+ ctx.Rule.NoDataState = models.NoDataSetAlerting
+ ctx.NoDataFound = true
+
+ ctx.Rule.State = ctx.GetNewState()
+ So(ctx.Rule.State, ShouldEqual, models.AlertStateAlerting)
+ })
+
+ Convey("ok -> no_data(keep_last)", func() {
+ ctx.PrevAlertState = models.AlertStateOK
+ ctx.Rule.NoDataState = models.NoDataKeepState
+ ctx.NoDataFound = true
+
+ ctx.Rule.State = ctx.GetNewState()
+ So(ctx.Rule.State, ShouldEqual, models.AlertStateOK)
+ })
+
+ Convey("pending -> no_data(keep_last)", func() {
+ ctx.PrevAlertState = models.AlertStatePending
+ ctx.Rule.NoDataState = models.NoDataKeepState
+ ctx.NoDataFound = true
+
+ ctx.Rule.State = ctx.GetNewState()
+ So(ctx.Rule.State, ShouldEqual, models.AlertStatePending)
+ })
+ })
})
}
diff --git a/pkg/services/alerting/eval_handler.go b/pkg/services/alerting/eval_handler.go
index 457e02000fa..aa24efa77cd 100644
--- a/pkg/services/alerting/eval_handler.go
+++ b/pkg/services/alerting/eval_handler.go
@@ -7,7 +7,6 @@ import (
"github.com/grafana/grafana/pkg/log"
"github.com/grafana/grafana/pkg/metrics"
- "github.com/grafana/grafana/pkg/models"
)
type DefaultEvalHandler struct {
@@ -66,40 +65,7 @@ func (e *DefaultEvalHandler) Eval(context *EvalContext) {
context.Firing = firing
context.NoDataFound = noDataFound
context.EndTime = time.Now()
- context.Rule.State = e.getNewState(context)
elapsedTime := context.EndTime.Sub(context.StartTime).Nanoseconds() / int64(time.Millisecond)
metrics.M_Alerting_Execution_Time.Observe(float64(elapsedTime))
}
-
-// This should be move into evalContext once its been refactored. (Carl Bergquist)
-func (handler *DefaultEvalHandler) getNewState(evalContext *EvalContext) models.AlertStateType {
- if evalContext.Error != nil {
- handler.log.Error("Alert Rule Result Error",
- "ruleId", evalContext.Rule.Id,
- "name", evalContext.Rule.Name,
- "error", evalContext.Error,
- "changing state to", evalContext.Rule.ExecutionErrorState.ToAlertState())
-
- if evalContext.Rule.ExecutionErrorState == models.ExecutionErrorKeepState {
- return evalContext.PrevAlertState
- } else {
- return evalContext.Rule.ExecutionErrorState.ToAlertState()
- }
- } else if evalContext.Firing {
- return models.AlertStateAlerting
- } else if evalContext.NoDataFound {
- handler.log.Info("Alert Rule returned no data",
- "ruleId", evalContext.Rule.Id,
- "name", evalContext.Rule.Name,
- "changing state to", evalContext.Rule.NoDataState.ToAlertState())
-
- if evalContext.Rule.NoDataState == models.NoDataKeepState {
- return evalContext.PrevAlertState
- } else {
- return evalContext.Rule.NoDataState.ToAlertState()
- }
- }
-
- return models.AlertStateOK
-}
diff --git a/pkg/services/alerting/eval_handler_test.go b/pkg/services/alerting/eval_handler_test.go
index c942e24818f..a7c1f1e67fa 100644
--- a/pkg/services/alerting/eval_handler_test.go
+++ b/pkg/services/alerting/eval_handler_test.go
@@ -2,10 +2,8 @@ package alerting
import (
"context"
- "fmt"
"testing"
- "github.com/grafana/grafana/pkg/models"
. "github.com/smartystreets/goconvey/convey"
)
@@ -203,73 +201,5 @@ func TestAlertingEvaluationHandler(t *testing.T) {
handler.Eval(context)
So(context.NoDataFound, ShouldBeTrue)
})
-
- Convey("EvalHandler can replace alert state based for errors and no_data", func() {
- ctx := NewEvalContext(context.TODO(), &Rule{Conditions: []Condition{&conditionStub{firing: true}}})
- dummieError := fmt.Errorf("dummie error")
- Convey("Should update alert state", func() {
-
- Convey("ok -> alerting", func() {
- ctx.PrevAlertState = models.AlertStateOK
- ctx.Firing = true
-
- So(handler.getNewState(ctx), ShouldEqual, models.AlertStateAlerting)
- })
-
- Convey("ok -> error(alerting)", func() {
- ctx.PrevAlertState = models.AlertStateOK
- ctx.Error = dummieError
- ctx.Rule.ExecutionErrorState = models.ExecutionErrorSetAlerting
-
- ctx.Rule.State = handler.getNewState(ctx)
- So(ctx.Rule.State, ShouldEqual, models.AlertStateAlerting)
- })
-
- Convey("ok -> error(keep_last)", func() {
- ctx.PrevAlertState = models.AlertStateOK
- ctx.Error = dummieError
- ctx.Rule.ExecutionErrorState = models.ExecutionErrorKeepState
-
- ctx.Rule.State = handler.getNewState(ctx)
- So(ctx.Rule.State, ShouldEqual, models.AlertStateOK)
- })
-
- Convey("pending -> error(keep_last)", func() {
- ctx.PrevAlertState = models.AlertStatePending
- ctx.Error = dummieError
- ctx.Rule.ExecutionErrorState = models.ExecutionErrorKeepState
-
- ctx.Rule.State = handler.getNewState(ctx)
- So(ctx.Rule.State, ShouldEqual, models.AlertStatePending)
- })
-
- Convey("ok -> no_data(alerting)", func() {
- ctx.PrevAlertState = models.AlertStateOK
- ctx.Rule.NoDataState = models.NoDataSetAlerting
- ctx.NoDataFound = true
-
- ctx.Rule.State = handler.getNewState(ctx)
- So(ctx.Rule.State, ShouldEqual, models.AlertStateAlerting)
- })
-
- Convey("ok -> no_data(keep_last)", func() {
- ctx.PrevAlertState = models.AlertStateOK
- ctx.Rule.NoDataState = models.NoDataKeepState
- ctx.NoDataFound = true
-
- ctx.Rule.State = handler.getNewState(ctx)
- So(ctx.Rule.State, ShouldEqual, models.AlertStateOK)
- })
-
- Convey("pending -> no_data(keep_last)", func() {
- ctx.PrevAlertState = models.AlertStatePending
- ctx.Rule.NoDataState = models.NoDataKeepState
- ctx.NoDataFound = true
-
- ctx.Rule.State = handler.getNewState(ctx)
- So(ctx.Rule.State, ShouldEqual, models.AlertStatePending)
- })
- })
- })
})
}
diff --git a/pkg/services/alerting/test_rule.go b/pkg/services/alerting/test_rule.go
index e3aa95e0ede..88418bff14e 100644
--- a/pkg/services/alerting/test_rule.go
+++ b/pkg/services/alerting/test_rule.go
@@ -53,6 +53,7 @@ func testAlertRule(rule *Rule) *EvalContext {
context.IsTestRun = true
handler.Eval(context)
+ context.Rule.State = context.GetNewState()
return context
}
diff --git a/pkg/services/dashboards/folder_service.go b/pkg/services/dashboards/folder_service.go
index 66afa6306fb..ae92952056e 100644
--- a/pkg/services/dashboards/folder_service.go
+++ b/pkg/services/dashboards/folder_service.go
@@ -10,8 +10,8 @@ import (
// FolderService service for operating on folders
type FolderService interface {
GetFolders(limit int) ([]*models.Folder, error)
- GetFolderById(id int64) (*models.Folder, error)
- GetFolderByUid(uid string) (*models.Folder, error)
+ GetFolderByID(id int64) (*models.Folder, error)
+ GetFolderByUID(uid string) (*models.Folder, error)
CreateFolder(cmd *models.CreateFolderCommand) error
UpdateFolder(uid string, cmd *models.UpdateFolderCommand) error
DeleteFolder(uid string) (*models.Folder, error)
@@ -57,7 +57,7 @@ func (dr *dashboardServiceImpl) GetFolders(limit int) ([]*models.Folder, error)
return folders, nil
}
-func (dr *dashboardServiceImpl) GetFolderById(id int64) (*models.Folder, error) {
+func (dr *dashboardServiceImpl) GetFolderByID(id int64) (*models.Folder, error) {
query := models.GetDashboardQuery{OrgId: dr.orgId, Id: id}
dashFolder, err := getFolder(query)
@@ -76,7 +76,7 @@ func (dr *dashboardServiceImpl) GetFolderById(id int64) (*models.Folder, error)
return dashToFolder(dashFolder), nil
}
-func (dr *dashboardServiceImpl) GetFolderByUid(uid string) (*models.Folder, error) {
+func (dr *dashboardServiceImpl) GetFolderByUID(uid string) (*models.Folder, error) {
query := models.GetDashboardQuery{OrgId: dr.orgId, Uid: uid}
dashFolder, err := getFolder(query)
diff --git a/pkg/services/dashboards/folder_service_test.go b/pkg/services/dashboards/folder_service_test.go
index 6357e84805a..6c0413d1878 100644
--- a/pkg/services/dashboards/folder_service_test.go
+++ b/pkg/services/dashboards/folder_service_test.go
@@ -36,13 +36,13 @@ func TestFolderService(t *testing.T) {
})
Convey("When get folder by id should return access denied error", func() {
- _, err := service.GetFolderById(1)
+ _, err := service.GetFolderByID(1)
So(err, ShouldNotBeNil)
So(err, ShouldEqual, models.ErrFolderAccessDenied)
})
Convey("When get folder by uid should return access denied error", func() {
- _, err := service.GetFolderByUid("uid")
+ _, err := service.GetFolderByUID("uid")
So(err, ShouldNotBeNil)
So(err, ShouldEqual, models.ErrFolderAccessDenied)
})
@@ -147,14 +147,14 @@ func TestFolderService(t *testing.T) {
})
Convey("When get folder by id should return folder", func() {
- f, _ := service.GetFolderById(1)
+ f, _ := service.GetFolderByID(1)
So(f.Id, ShouldEqual, dashFolder.Id)
So(f.Uid, ShouldEqual, dashFolder.Uid)
So(f.Title, ShouldEqual, dashFolder.Title)
})
Convey("When get folder by uid should return folder", func() {
- f, _ := service.GetFolderByUid("uid")
+ f, _ := service.GetFolderByUID("uid")
So(f.Id, ShouldEqual, dashFolder.Id)
So(f.Uid, ShouldEqual, dashFolder.Uid)
So(f.Title, ShouldEqual, dashFolder.Title)
diff --git a/pkg/services/sqlstore/dashboard_version.go b/pkg/services/sqlstore/dashboard_version.go
index 547f62628f3..1f2850b2021 100644
--- a/pkg/services/sqlstore/dashboard_version.go
+++ b/pkg/services/sqlstore/dashboard_version.go
@@ -67,30 +67,39 @@ func GetDashboardVersions(query *m.GetDashboardVersionsQuery) error {
return nil
}
+const MAX_VERSIONS_TO_DELETE = 100
+
func DeleteExpiredVersions(cmd *m.DeleteExpiredVersionsCommand) error {
return inTransaction(func(sess *DBSession) error {
- versions := []DashboardVersionExp{}
versionsToKeep := setting.DashboardVersionsToKeep
-
if versionsToKeep < 1 {
versionsToKeep = 1
}
- err := sess.Table("dashboard_version").
- Select("dashboard_version.id, dashboard_version.version, dashboard_version.dashboard_id").
- Where(`dashboard_id IN (
- SELECT dashboard_id FROM dashboard_version
- GROUP BY dashboard_id HAVING COUNT(dashboard_version.id) > ?
- )`, versionsToKeep).
- Desc("dashboard_version.dashboard_id", "dashboard_version.version").
- Find(&versions)
+ // Idea of this query is finding version IDs to delete based on formula:
+ // min_version_to_keep = min_version + (versions_count - versions_to_keep)
+ // where version stats is processed for each dashboard. This guarantees that we keep at least versions_to_keep
+ // versions, but in some cases (when versions are sparse) this number may be more.
+ versionIdsToDeleteQuery := `SELECT id
+ FROM dashboard_version, (
+ SELECT dashboard_id, count(version) as count, min(version) as min
+ FROM dashboard_version
+ GROUP BY dashboard_id
+ ) AS vtd
+ WHERE dashboard_version.dashboard_id=vtd.dashboard_id
+ AND version < vtd.min + vtd.count - ?`
+ var versionIdsToDelete []interface{}
+ err := sess.SQL(versionIdsToDeleteQuery, versionsToKeep).Find(&versionIdsToDelete)
if err != nil {
return err
}
- // Keep last versionsToKeep versions and delete other
- versionIdsToDelete := getVersionIDsToDelete(versions, versionsToKeep)
+ // Don't delete more than MAX_VERSIONS_TO_DELETE version per time
+ if len(versionIdsToDelete) > MAX_VERSIONS_TO_DELETE {
+ versionIdsToDelete = versionIdsToDelete[:MAX_VERSIONS_TO_DELETE]
+ }
+
if len(versionIdsToDelete) > 0 {
deleteExpiredSql := `DELETE FROM dashboard_version WHERE id IN (?` + strings.Repeat(",?", len(versionIdsToDelete)-1) + `)`
expiredResponse, err := sess.Exec(deleteExpiredSql, versionIdsToDelete...)
@@ -103,34 +112,3 @@ func DeleteExpiredVersions(cmd *m.DeleteExpiredVersionsCommand) error {
return nil
})
}
-
-// Short version of DashboardVersion for getting expired versions
-type DashboardVersionExp struct {
- Id int64 `json:"id"`
- DashboardId int64 `json:"dashboardId"`
- Version int `json:"version"`
-}
-
-func getVersionIDsToDelete(versions []DashboardVersionExp, versionsToKeep int) []interface{} {
- versionIds := make([]interface{}, 0)
-
- if len(versions) == 0 {
- return versionIds
- }
-
- currentDashboard := versions[0].DashboardId
- count := 0
- for _, v := range versions {
- if v.DashboardId == currentDashboard {
- count++
- } else {
- count = 1
- currentDashboard = v.DashboardId
- }
- if count > versionsToKeep {
- versionIds = append(versionIds, v.Id)
- }
- }
-
- return versionIds
-}
diff --git a/pkg/services/sqlstore/dashboard_version_test.go b/pkg/services/sqlstore/dashboard_version_test.go
index 1b74e7847c4..a6403755d05 100644
--- a/pkg/services/sqlstore/dashboard_version_test.go
+++ b/pkg/services/sqlstore/dashboard_version_test.go
@@ -136,10 +136,30 @@ func TestDeleteExpiredVersions(t *testing.T) {
err := DeleteExpiredVersions(&m.DeleteExpiredVersionsCommand{})
So(err, ShouldBeNil)
- query := m.GetDashboardVersionsQuery{DashboardId: savedDash.Id, OrgId: 1}
+ query := m.GetDashboardVersionsQuery{DashboardId: savedDash.Id, OrgId: 1, Limit: versionsToWrite}
GetDashboardVersions(&query)
So(len(query.Result), ShouldEqual, versionsToWrite)
})
+
+ Convey("Don't delete more than MAX_VERSIONS_TO_DELETE per iteration", func() {
+ versionsToWriteBigNumber := MAX_VERSIONS_TO_DELETE + versionsToWrite
+ for i := 0; i < versionsToWriteBigNumber-versionsToWrite; i++ {
+ updateTestDashboard(savedDash, map[string]interface{}{
+ "tags": "different-tag",
+ })
+ }
+
+ err := DeleteExpiredVersions(&m.DeleteExpiredVersionsCommand{})
+ So(err, ShouldBeNil)
+
+ query := m.GetDashboardVersionsQuery{DashboardId: savedDash.Id, OrgId: 1, Limit: versionsToWriteBigNumber}
+ GetDashboardVersions(&query)
+
+ // Ensure we have at least versionsToKeep versions
+ So(len(query.Result), ShouldBeGreaterThanOrEqualTo, versionsToKeep)
+ // Ensure we haven't deleted more than MAX_VERSIONS_TO_DELETE rows
+ So(versionsToWriteBigNumber-len(query.Result), ShouldBeLessThanOrEqualTo, MAX_VERSIONS_TO_DELETE)
+ })
})
}
diff --git a/pkg/services/sqlstore/user.go b/pkg/services/sqlstore/user.go
index 73ea07f031f..f42ff5fb2ed 100644
--- a/pkg/services/sqlstore/user.go
+++ b/pkg/services/sqlstore/user.go
@@ -315,6 +315,7 @@ func GetUserProfile(query *m.GetUserProfileQuery) error {
}
query.Result = m.UserProfileDTO{
+ Id: user.Id,
Name: user.Name,
Email: user.Email,
Login: user.Login,
diff --git a/pkg/tsdb/mssql/macros.go b/pkg/tsdb/mssql/macros.go
index 92c6ede148e..9d41cd03255 100644
--- a/pkg/tsdb/mssql/macros.go
+++ b/pkg/tsdb/mssql/macros.go
@@ -73,25 +73,20 @@ func (m *MsSqlMacroEngine) evaluateMacro(name string, args []string) (string, er
return "", fmt.Errorf("missing time column argument for macro %v", name)
}
return fmt.Sprintf("%s AS time", args[0]), nil
- case "__utcTime":
- if len(args) == 0 {
- return "", fmt.Errorf("missing time column argument for macro %v", name)
- }
- return fmt.Sprintf("DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), %s) AS time", args[0]), nil
case "__timeEpoch":
if len(args) == 0 {
return "", fmt.Errorf("missing time column argument for macro %v", name)
}
- return fmt.Sprintf("DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), %s) ) AS time", args[0]), nil
+ return fmt.Sprintf("DATEDIFF(second, '1970-01-01', %s) AS time", args[0]), nil
case "__timeFilter":
if len(args) == 0 {
return "", fmt.Errorf("missing time column argument for macro %v", name)
}
- return fmt.Sprintf("%s >= DATEADD(s, %d+DATEDIFF(second,GETUTCDATE(),GETDATE()), '1970-01-01') AND %s <= DATEADD(s, %d+DATEDIFF(second,GETUTCDATE(),GETDATE()), '1970-01-01')", args[0], uint64(m.TimeRange.GetFromAsMsEpoch()/1000), args[0], uint64(m.TimeRange.GetToAsMsEpoch()/1000)), nil
+ return fmt.Sprintf("%s >= DATEADD(s, %d, '1970-01-01') AND %s <= DATEADD(s, %d, '1970-01-01')", args[0], uint64(m.TimeRange.GetFromAsMsEpoch()/1000), args[0], uint64(m.TimeRange.GetToAsMsEpoch()/1000)), nil
case "__timeFrom":
- return fmt.Sprintf("DATEADD(second, %d+DATEDIFF(second,GETUTCDATE(),GETDATE()), '1970-01-01')", uint64(m.TimeRange.GetFromAsMsEpoch()/1000)), nil
+ return fmt.Sprintf("DATEADD(second, %d, '1970-01-01')", uint64(m.TimeRange.GetFromAsMsEpoch()/1000)), nil
case "__timeTo":
- return fmt.Sprintf("DATEADD(second, %d+DATEDIFF(second,GETUTCDATE(),GETDATE()), '1970-01-01')", uint64(m.TimeRange.GetToAsMsEpoch()/1000)), nil
+ return fmt.Sprintf("DATEADD(second, %d, '1970-01-01')", uint64(m.TimeRange.GetToAsMsEpoch()/1000)), nil
case "__timeGroup":
if len(args) < 2 {
return "", fmt.Errorf("macro %v needs time column and interval", name)
@@ -113,7 +108,7 @@ func (m *MsSqlMacroEngine) evaluateMacro(name string, args []string) (string, er
m.Query.Model.Set("fillValue", floatVal)
}
}
- return fmt.Sprintf("cast(cast(DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), %s))/%.0f as int)*%.0f as int)", args[0], interval.Seconds(), interval.Seconds()), nil
+ return fmt.Sprintf("CAST(ROUND(DATEDIFF(second, '1970-01-01', %s)/%.1f, 0) as bigint)*%.0f", args[0], interval.Seconds(), interval.Seconds()), nil
case "__unixEpochFilter":
if len(args) == 0 {
return "", fmt.Errorf("missing time column argument for macro %v", name)
diff --git a/pkg/tsdb/mssql/macros_test.go b/pkg/tsdb/mssql/macros_test.go
index db1f5670924..12a9b0d82be 100644
--- a/pkg/tsdb/mssql/macros_test.go
+++ b/pkg/tsdb/mssql/macros_test.go
@@ -25,46 +25,39 @@ func TestMacroEngine(t *testing.T) {
So(sql, ShouldEqual, "select time_column AS time")
})
- Convey("interpolate __utcTime function", func() {
- sql, err := engine.Interpolate(query, nil, "select $__utcTime(time_column)")
- So(err, ShouldBeNil)
-
- So(sql, ShouldEqual, "select DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), time_column) AS time")
- })
-
Convey("interpolate __timeEpoch function", func() {
sql, err := engine.Interpolate(query, nil, "select $__timeEpoch(time_column)")
So(err, ShouldBeNil)
- So(sql, ShouldEqual, "select DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), time_column) ) AS time")
+ So(sql, ShouldEqual, "select DATEDIFF(second, '1970-01-01', time_column) AS time")
})
Convey("interpolate __timeEpoch function wrapped in aggregation", func() {
sql, err := engine.Interpolate(query, nil, "select min($__timeEpoch(time_column))")
So(err, ShouldBeNil)
- So(sql, ShouldEqual, "select min(DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), time_column) ) AS time)")
+ So(sql, ShouldEqual, "select min(DATEDIFF(second, '1970-01-01', time_column) AS time)")
})
Convey("interpolate __timeFilter function", func() {
sql, err := engine.Interpolate(query, timeRange, "WHERE $__timeFilter(time_column)")
So(err, ShouldBeNil)
- So(sql, ShouldEqual, "WHERE time_column >= DATEADD(s, 18446744066914186738+DATEDIFF(second,GETUTCDATE(),GETDATE()), '1970-01-01') AND time_column <= DATEADD(s, 18446744066914187038+DATEDIFF(second,GETUTCDATE(),GETDATE()), '1970-01-01')")
+ So(sql, ShouldEqual, "WHERE time_column >= DATEADD(s, 18446744066914186738, '1970-01-01') AND time_column <= DATEADD(s, 18446744066914187038, '1970-01-01')")
})
Convey("interpolate __timeGroup function", func() {
sql, err := engine.Interpolate(query, timeRange, "GROUP BY $__timeGroup(time_column,'5m')")
So(err, ShouldBeNil)
- So(sql, ShouldEqual, "GROUP BY cast(cast(DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), time_column))/300 as int)*300 as int)")
+ So(sql, ShouldEqual, "GROUP BY CAST(ROUND(DATEDIFF(second, '1970-01-01', time_column)/300.0, 0) as bigint)*300")
})
Convey("interpolate __timeGroup function with spaces around arguments", func() {
sql, err := engine.Interpolate(query, timeRange, "GROUP BY $__timeGroup(time_column , '5m')")
So(err, ShouldBeNil)
- So(sql, ShouldEqual, "GROUP BY cast(cast(DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), time_column))/300 as int)*300 as int)")
+ So(sql, ShouldEqual, "GROUP BY CAST(ROUND(DATEDIFF(second, '1970-01-01', time_column)/300.0, 0) as bigint)*300")
})
Convey("interpolate __timeGroup function with fill (value = NULL)", func() {
@@ -97,21 +90,21 @@ func TestMacroEngine(t *testing.T) {
sql, err := engine.Interpolate(query, timeRange, "select $__timeFrom(time_column)")
So(err, ShouldBeNil)
- So(sql, ShouldEqual, "select DATEADD(second, 18446744066914186738+DATEDIFF(second,GETUTCDATE(),GETDATE()), '1970-01-01')")
+ So(sql, ShouldEqual, "select DATEADD(second, 18446744066914186738, '1970-01-01')")
})
Convey("interpolate __timeTo function", func() {
sql, err := engine.Interpolate(query, timeRange, "select $__timeTo(time_column)")
So(err, ShouldBeNil)
- So(sql, ShouldEqual, "select DATEADD(second, 18446744066914187038+DATEDIFF(second,GETUTCDATE(),GETDATE()), '1970-01-01')")
+ So(sql, ShouldEqual, "select DATEADD(second, 18446744066914187038, '1970-01-01')")
})
Convey("interpolate __unixEpochFilter function", func() {
- sql, err := engine.Interpolate(query, timeRange, "select $__unixEpochFilter(18446744066914186738)")
+ sql, err := engine.Interpolate(query, timeRange, "select $__unixEpochFilter(time_column)")
So(err, ShouldBeNil)
- So(sql, ShouldEqual, "select 18446744066914186738 >= 18446744066914186738 AND 18446744066914186738 <= 18446744066914187038")
+ So(sql, ShouldEqual, "select time_column >= 18446744066914186738 AND time_column <= 18446744066914187038")
})
Convey("interpolate __unixEpochFrom function", func() {
diff --git a/pkg/tsdb/mssql/mssql.go b/pkg/tsdb/mssql/mssql.go
index af68ca0424e..2638fd8bb40 100644
--- a/pkg/tsdb/mssql/mssql.go
+++ b/pkg/tsdb/mssql/mssql.go
@@ -119,15 +119,10 @@ func (e MssqlQueryEndpoint) transformToTable(query *tsdb.Query, rows *core.Rows,
return err
}
- // convert column named time to unix timestamp to make
- // native datetime mssql types work in annotation queries
- if timeIndex != -1 {
- switch value := values[timeIndex].(type) {
- case time.Time:
- values[timeIndex] = float64(value.Unix())
- }
- }
-
+ // converts column named time to unix timestamp in milliseconds
+ // to make native mssql datetime types and epoch dates work in
+ // annotation and table queries.
+ tsdb.ConvertSqlTimeColumnToEpochMs(values, timeIndex)
table.Rows = append(table.Rows, values)
}
diff --git a/pkg/tsdb/mssql/mssql_test.go b/pkg/tsdb/mssql/mssql_test.go
index 88b35b1aa2c..4bd1e3a8ad7 100644
--- a/pkg/tsdb/mssql/mssql_test.go
+++ b/pkg/tsdb/mssql/mssql_test.go
@@ -19,6 +19,8 @@ import (
// and set up a MSSQL db named grafanatest and a user/password grafana/Password!
// Use the docker/blocks/mssql_tests/docker-compose.yaml to spin up a
// preconfigured MSSQL server suitable for running these tests.
+// Thers's also a dashboard.json in same directory that you can import to Grafana
+// once you've created a datasource for the test server/database.
// If needed, change the variable below to the IP address of the database.
var serverIP string = "localhost"
@@ -37,44 +39,44 @@ func TestMSSQL(t *testing.T) {
sess := x.NewSession()
defer sess.Close()
- fromStart := time.Date(2018, 3, 15, 13, 0, 0, 0, time.UTC)
+ fromStart := time.Date(2018, 3, 15, 13, 0, 0, 0, time.UTC).In(time.Local)
Convey("Given a table with different native data types", func() {
sql := `
- IF OBJECT_ID('dbo.[mssql_types]', 'U') IS NOT NULL
- DROP TABLE dbo.[mssql_types]
+ IF OBJECT_ID('dbo.[mssql_types]', 'U') IS NOT NULL
+ DROP TABLE dbo.[mssql_types]
- CREATE TABLE [mssql_types] (
- c_bit bit,
+ CREATE TABLE [mssql_types] (
+ c_bit bit,
- c_tinyint tinyint,
- c_smallint smallint,
- c_int int,
- c_bigint bigint,
+ c_tinyint tinyint,
+ c_smallint smallint,
+ c_int int,
+ c_bigint bigint,
- c_money money,
- c_smallmoney smallmoney,
- c_numeric numeric(10,5),
- c_real real,
- c_decimal decimal(10,2),
- c_float float,
+ c_money money,
+ c_smallmoney smallmoney,
+ c_numeric numeric(10,5),
+ c_real real,
+ c_decimal decimal(10,2),
+ c_float float,
- c_char char(10),
- c_varchar varchar(10),
- c_text text,
+ c_char char(10),
+ c_varchar varchar(10),
+ c_text text,
- c_nchar nchar(12),
- c_nvarchar nvarchar(12),
- c_ntext ntext,
+ c_nchar nchar(12),
+ c_nvarchar nvarchar(12),
+ c_ntext ntext,
- c_datetime datetime,
- c_datetime2 datetime2,
- c_smalldatetime smalldatetime,
- c_date date,
- c_time time,
- c_datetimeoffset datetimeoffset
- )
- `
+ c_datetime datetime,
+ c_datetime2 datetime2,
+ c_smalldatetime smalldatetime,
+ c_date date,
+ c_time time,
+ c_datetimeoffset datetimeoffset
+ )
+ `
_, err := sess.Exec(sql)
So(err, ShouldBeNil)
@@ -87,14 +89,14 @@ func TestMSSQL(t *testing.T) {
d2 := dt2.Format(dt2Format)
sql = fmt.Sprintf(`
- INSERT INTO [mssql_types]
- SELECT
- 1, 5, 20020, 980300, 1420070400, '$20000.15', '£2.15', 12345.12,
- 1.11, 2.22, 3.33,
- 'char10', 'varchar10', 'text',
- N'☺nchar12☺', N'☺nvarchar12☺', N'☺text☺',
- CAST('%s' AS DATETIME), CAST('%s' AS DATETIME2), CAST('%s' AS SMALLDATETIME), CAST('%s' AS DATE), CAST('%s' AS TIME), SWITCHOFFSET(CAST('%s' AS DATETIMEOFFSET), '-07:00')
- `, d, d2, d, d, d, d2)
+ INSERT INTO [mssql_types]
+ SELECT
+ 1, 5, 20020, 980300, 1420070400, '$20000.15', '£2.15', 12345.12,
+ 1.11, 2.22, 3.33,
+ 'char10', 'varchar10', 'text',
+ N'☺nchar12☺', N'☺nvarchar12☺', N'☺text☺',
+ CAST('%s' AS DATETIME), CAST('%s' AS DATETIME2), CAST('%s' AS SMALLDATETIME), CAST('%s' AS DATE), CAST('%s' AS TIME), SWITCHOFFSET(CAST('%s' AS DATETIMEOFFSET), '-07:00')
+ `, d, d2, d, d, d, d2)
_, err = sess.Exec(sql)
So(err, ShouldBeNil)
@@ -151,14 +153,14 @@ func TestMSSQL(t *testing.T) {
Convey("Given a table with metrics that lacks data for some series ", func() {
sql := `
- IF OBJECT_ID('dbo.[metric]', 'U') IS NOT NULL
- DROP TABLE dbo.[metric]
+ IF OBJECT_ID('dbo.[metric]', 'U') IS NOT NULL
+ DROP TABLE dbo.[metric]
- CREATE TABLE [metric] (
- time datetime,
- value int
- )
- `
+ CREATE TABLE [metric] (
+ time datetime,
+ value int
+ )
+ `
_, err := sess.Exec(sql)
So(err, ShouldBeNil)
@@ -186,14 +188,8 @@ func TestMSSQL(t *testing.T) {
})
}
- dtFormat := "2006-01-02 15:04:05.999999999"
for _, s := range series {
- sql = fmt.Sprintf(`
- INSERT INTO metric (time, value)
- VALUES(CAST('%s' AS DATETIME), %d)
- `, s.Time.Format(dtFormat), s.Value)
-
- _, err = sess.Exec(sql)
+ _, err = sess.Insert(s)
So(err, ShouldBeNil)
}
@@ -211,22 +207,32 @@ func TestMSSQL(t *testing.T) {
}
resp, err := endpoint.Query(nil, nil, query)
- queryResult := resp.Results["A"]
So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
So(queryResult.Error, ShouldBeNil)
points := queryResult.Series[0].Points
+ So(len(points), ShouldEqual, 6)
- So(len(points), ShouldEqual, 4)
- actualValueFirst := points[0][0].Float64
- actualTimeFirst := time.Unix(int64(points[0][1].Float64)/1000, 0)
- So(actualValueFirst, ShouldEqual, 15)
- So(actualTimeFirst, ShouldEqual, fromStart)
+ dt := fromStart
- actualValueLast := points[3][0].Float64
- actualTimeLast := time.Unix(int64(points[3][1].Float64)/1000, 0)
- So(actualValueLast, ShouldEqual, 20)
- So(actualTimeLast, ShouldEqual, fromStart.Add(25*time.Minute))
+ for i := 0; i < 3; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 15)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
+
+ // adjust for 5 minute gap
+ dt = dt.Add(5 * time.Minute)
+ for i := 3; i < 6; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 20)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
})
Convey("When doing a metric query using timeGroup with NULL fill enabled", func() {
@@ -247,33 +253,34 @@ func TestMSSQL(t *testing.T) {
}
resp, err := endpoint.Query(nil, nil, query)
- queryResult := resp.Results["A"]
So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
So(queryResult.Error, ShouldBeNil)
points := queryResult.Series[0].Points
-
So(len(points), ShouldEqual, 7)
- actualValueFirst := points[0][0].Float64
- actualTimeFirst := time.Unix(int64(points[0][1].Float64)/1000, 0)
- So(actualValueFirst, ShouldEqual, 15)
- So(actualTimeFirst, ShouldEqual, fromStart)
- actualNullPoint := points[3][0]
- actualNullTime := time.Unix(int64(points[3][1].Float64)/1000, 0)
- So(actualNullPoint.Valid, ShouldBeFalse)
- So(actualNullTime, ShouldEqual, fromStart.Add(15*time.Minute))
+ dt := fromStart
- actualValueLast := points[5][0].Float64
- actualTimeLast := time.Unix(int64(points[5][1].Float64)/1000, 0)
- So(actualValueLast, ShouldEqual, 20)
- So(actualTimeLast, ShouldEqual, fromStart.Add(25*time.Minute))
+ for i := 0; i < 3; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 15)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
- actualLastNullPoint := points[6][0]
- actualLastNullTime := time.Unix(int64(points[6][1].Float64)/1000, 0)
- So(actualLastNullPoint.Valid, ShouldBeFalse)
- So(actualLastNullTime, ShouldEqual, fromStart.Add(30*time.Minute))
+ So(points[3][0].Valid, ShouldBeFalse)
+ // adjust for 5 minute gap
+ dt = dt.Add(5 * time.Minute)
+ for i := 4; i < 7; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 20)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
})
Convey("When doing a metric query using timeGroup with float fill enabled", func() {
@@ -294,53 +301,44 @@ func TestMSSQL(t *testing.T) {
}
resp, err := endpoint.Query(nil, nil, query)
- queryResult := resp.Results["A"]
So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
So(queryResult.Error, ShouldBeNil)
points := queryResult.Series[0].Points
-
- So(points[6][0].Float64, ShouldEqual, 1.5)
+ So(points[3][0].Float64, ShouldEqual, 1.5)
})
})
Convey("Given a table with metrics having multiple values and measurements", func() {
- sql := `
- IF OBJECT_ID('dbo.[metric_values]', 'U') IS NOT NULL
- DROP TABLE dbo.[metric_values]
-
- CREATE TABLE [metric_values] (
- time datetime,
- measurement nvarchar(100),
- valueOne int,
- valueTwo int,
- )
- `
-
- _, err := sess.Exec(sql)
- So(err, ShouldBeNil)
-
- type metricValues struct {
+ type metric_values struct {
Time time.Time
Measurement string
- ValueOne int64
- ValueTwo int64
+ ValueOne int64 `xorm:"integer 'valueOne'"`
+ ValueTwo int64 `xorm:"integer 'valueTwo'"`
}
+ if exist, err := sess.IsTableExist(metric_values{}); err != nil || exist {
+ So(err, ShouldBeNil)
+ sess.DropTable(metric_values{})
+ }
+ err := sess.CreateTable(metric_values{})
+ So(err, ShouldBeNil)
+
rand.Seed(time.Now().Unix())
rnd := func(min, max int64) int64 {
return rand.Int63n(max-min) + min
}
- series := []*metricValues{}
+ series := []*metric_values{}
for _, t := range genTimeRangeByInterval(fromStart.Add(-30*time.Minute), 90*time.Minute, 5*time.Minute) {
- series = append(series, &metricValues{
+ series = append(series, &metric_values{
Time: t,
Measurement: "Metric A",
ValueOne: rnd(0, 100),
ValueTwo: rnd(0, 100),
})
- series = append(series, &metricValues{
+ series = append(series, &metric_values{
Time: t,
Measurement: "Metric B",
ValueOne: rnd(0, 100),
@@ -348,14 +346,8 @@ func TestMSSQL(t *testing.T) {
})
}
- dtFormat := "2006-01-02 15:04:05"
for _, s := range series {
- sql = fmt.Sprintf(`
- INSERT metric_values (time, measurement, valueOne, valueTwo)
- VALUES(CAST('%s' AS DATETIME), '%s', %d, %d)
- `, s.Time.Format(dtFormat), s.Measurement, s.ValueOne, s.ValueTwo)
-
- _, err = sess.Exec(sql)
+ _, err = sess.Insert(s)
So(err, ShouldBeNil)
}
@@ -373,8 +365,8 @@ func TestMSSQL(t *testing.T) {
}
resp, err := endpoint.Query(nil, nil, query)
- queryResult := resp.Results["A"]
So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
So(queryResult.Error, ShouldBeNil)
So(len(queryResult.Series), ShouldEqual, 2)
@@ -396,8 +388,8 @@ func TestMSSQL(t *testing.T) {
}
resp, err := endpoint.Query(nil, nil, query)
- queryResult := resp.Results["A"]
So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
So(queryResult.Error, ShouldBeNil)
So(len(queryResult.Series), ShouldEqual, 2)
@@ -407,45 +399,55 @@ func TestMSSQL(t *testing.T) {
Convey("Given a stored procedure that takes @from and @to in epoch time", func() {
sql := `
- IF object_id('sp_test_epoch') IS NOT NULL
- DROP PROCEDURE sp_test_epoch
- `
+ IF object_id('sp_test_epoch') IS NOT NULL
+ DROP PROCEDURE sp_test_epoch
+ `
_, err := sess.Exec(sql)
So(err, ShouldBeNil)
sql = `
- CREATE PROCEDURE sp_test_epoch(
- @from int,
- @to int
- ) AS
- BEGIN
- SELECT
- cast(cast(DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), time))/600 as int)*600 as int) as time,
- measurement + ' - value one' as metric,
- avg(valueOne) as value
- FROM
- metric_values
- WHERE
- time >= DATEADD(s, @from, '1970-01-01') AND time <= DATEADD(s, @to, '1970-01-01')
- GROUP BY
- cast(cast(DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), time))/600 as int)*600 as int),
- measurement
- UNION ALL
- SELECT
- cast(cast(DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), time))/600 as int)*600 as int) as time,
- measurement + ' - value two' as metric,
- avg(valueTwo) as value
- FROM
- metric_values
- WHERE
- time >= DATEADD(s, @from, '1970-01-01') AND time <= DATEADD(s, @to, '1970-01-01')
- GROUP BY
- cast(cast(DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second,GETDATE(),GETUTCDATE()), time))/600 as int)*600 as int),
- measurement
- ORDER BY 1
- END
- `
+ CREATE PROCEDURE sp_test_epoch(
+ @from int,
+ @to int,
+ @interval nvarchar(50) = '5m',
+ @metric nvarchar(200) = 'ALL'
+ ) AS
+ BEGIN
+ DECLARE @dInterval int
+ SELECT @dInterval = 300
+
+ IF @interval = '10m'
+ SELECT @dInterval = 600
+
+ SELECT
+ CAST(ROUND(DATEDIFF(second, '1970-01-01', time)/CAST(@dInterval as float), 0) as bigint)*@dInterval as time,
+ measurement + ' - value one' as metric,
+ avg(valueOne) as value
+ FROM
+ metric_values
+ WHERE
+ time BETWEEN DATEADD(s, @from, '1970-01-01') AND DATEADD(s, @to, '1970-01-01') AND
+ (@metric = 'ALL' OR measurement = @metric)
+ GROUP BY
+ CAST(ROUND(DATEDIFF(second, '1970-01-01', time)/CAST(@dInterval as float), 0) as bigint)*@dInterval,
+ measurement
+ UNION ALL
+ SELECT
+ CAST(ROUND(DATEDIFF(second, '1970-01-01', time)/CAST(@dInterval as float), 0) as bigint)*@dInterval as time,
+ measurement + ' - value two' as metric,
+ avg(valueTwo) as value
+ FROM
+ metric_values
+ WHERE
+ time BETWEEN DATEADD(s, @from, '1970-01-01') AND DATEADD(s, @to, '1970-01-01') AND
+ (@metric = 'ALL' OR measurement = @metric)
+ GROUP BY
+ CAST(ROUND(DATEDIFF(second, '1970-01-01', time)/CAST(@dInterval as float), 0) as bigint)*@dInterval,
+ measurement
+ ORDER BY 1
+ END
+ `
_, err = sess.Exec(sql)
So(err, ShouldBeNil)
@@ -456,10 +458,10 @@ func TestMSSQL(t *testing.T) {
{
Model: simplejson.NewFromAny(map[string]interface{}{
"rawSql": `DECLARE
- @from int = $__unixEpochFrom(),
- @to int = $__unixEpochTo()
+ @from int = $__unixEpochFrom(),
+ @to int = $__unixEpochTo()
- EXEC dbo.sp_test_epoch @from, @to`,
+ EXEC dbo.sp_test_epoch @from, @to`,
"format": "time_series",
}),
RefId: "A",
@@ -474,6 +476,7 @@ func TestMSSQL(t *testing.T) {
resp, err := endpoint.Query(nil, nil, query)
queryResult := resp.Results["A"]
So(err, ShouldBeNil)
+ fmt.Println("query", "sql", queryResult.Meta)
So(queryResult.Error, ShouldBeNil)
So(len(queryResult.Series), ShouldEqual, 4)
@@ -486,45 +489,55 @@ func TestMSSQL(t *testing.T) {
Convey("Given a stored procedure that takes @from and @to in datetime", func() {
sql := `
- IF object_id('sp_test_datetime') IS NOT NULL
- DROP PROCEDURE sp_test_datetime
- `
+ IF object_id('sp_test_datetime') IS NOT NULL
+ DROP PROCEDURE sp_test_datetime
+ `
_, err := sess.Exec(sql)
So(err, ShouldBeNil)
sql = `
- CREATE PROCEDURE sp_test_datetime(
- @from datetime,
- @to datetime
- ) AS
- BEGIN
- SELECT
- cast(cast(DATEDIFF(second, {d '1970-01-01'}, time)/600 as int)*600 as int) as time,
- measurement + ' - value one' as metric,
- avg(valueOne) as value
- FROM
- metric_values
- WHERE
- time >= @from AND time <= @to
- GROUP BY
- cast(cast(DATEDIFF(second, {d '1970-01-01'}, time)/600 as int)*600 as int),
- measurement
- UNION ALL
- SELECT
- cast(cast(DATEDIFF(second, {d '1970-01-01'}, time)/600 as int)*600 as int) as time,
- measurement + ' - value two' as metric,
- avg(valueTwo) as value
- FROM
- metric_values
- WHERE
- time >= @from AND time <= @to
- GROUP BY
- cast(cast(DATEDIFF(second, {d '1970-01-01'}, time)/600 as int)*600 as int),
- measurement
- ORDER BY 1
- END
- `
+ CREATE PROCEDURE sp_test_datetime(
+ @from datetime,
+ @to datetime,
+ @interval nvarchar(50) = '5m',
+ @metric nvarchar(200) = 'ALL'
+ ) AS
+ BEGIN
+ DECLARE @dInterval int
+ SELECT @dInterval = 300
+
+ IF @interval = '10m'
+ SELECT @dInterval = 600
+
+ SELECT
+ CAST(ROUND(DATEDIFF(second, '1970-01-01', time)/CAST(@dInterval as float), 0) as bigint)*@dInterval as time,
+ measurement + ' - value one' as metric,
+ avg(valueOne) as value
+ FROM
+ metric_values
+ WHERE
+ time BETWEEN @from AND @to AND
+ (@metric = 'ALL' OR measurement = @metric)
+ GROUP BY
+ CAST(ROUND(DATEDIFF(second, '1970-01-01', time)/CAST(@dInterval as float), 0) as bigint)*@dInterval,
+ measurement
+ UNION ALL
+ SELECT
+ CAST(ROUND(DATEDIFF(second, '1970-01-01', time)/CAST(@dInterval as float), 0) as bigint)*@dInterval as time,
+ measurement + ' - value two' as metric,
+ avg(valueTwo) as value
+ FROM
+ metric_values
+ WHERE
+ time BETWEEN @from AND @to AND
+ (@metric = 'ALL' OR measurement = @metric)
+ GROUP BY
+ CAST(ROUND(DATEDIFF(second, '1970-01-01', time)/CAST(@dInterval as float), 0) as bigint)*@dInterval,
+ measurement
+ ORDER BY 1
+ END
+ `
_, err = sess.Exec(sql)
So(err, ShouldBeNil)
@@ -535,10 +548,10 @@ func TestMSSQL(t *testing.T) {
{
Model: simplejson.NewFromAny(map[string]interface{}{
"rawSql": `DECLARE
- @from int = $__unixEpochFrom(),
- @to int = $__unixEpochTo()
+ @from int = $__unixEpochFrom(),
+ @to int = $__unixEpochTo()
- EXEC dbo.sp_test_epoch @from, @to`,
+ EXEC dbo.sp_test_epoch @from, @to`,
"format": "time_series",
}),
RefId: "A",
@@ -570,7 +583,7 @@ func TestMSSQL(t *testing.T) {
DROP TABLE dbo.[event]
CREATE TABLE [event] (
- time_sec bigint,
+ time_sec int,
description nvarchar(100),
tags nvarchar(100),
)
@@ -654,12 +667,191 @@ func TestMSSQL(t *testing.T) {
So(err, ShouldBeNil)
So(len(queryResult.Tables[0].Rows), ShouldEqual, 3)
})
+
+ Convey("When doing an annotation query with a time column in datetime format", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 527e6, time.UTC)
+ dtFormat := "2006-01-02 15:04:05.999999999"
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ CAST('%s' AS DATETIME) as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Format(dtFormat)),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(float64), ShouldEqual, float64(dt.Unix()*1000))
+ })
+
+ Convey("When doing an annotation query with a time column in epoch second format should return ms", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 527e6, time.UTC)
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ %d as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Unix()),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(int64), ShouldEqual, int64(dt.Unix()*1000))
+ })
+
+ Convey("When doing an annotation query with a time column in epoch second format (int) should return ms", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 527e6, time.UTC)
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ cast(%d as int) as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Unix()),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(int64), ShouldEqual, int64(dt.Unix()*1000))
+ })
+
+ Convey("When doing an annotation query with a time column in epoch millisecond format should return ms", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 527e6, time.UTC)
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ %d as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Unix()*1000),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(float64), ShouldEqual, float64(dt.Unix()*1000))
+ })
+
+ Convey("When doing an annotation query with a time column holding a bigint null value should return nil", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT
+ cast(null as bigint) as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `,
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0], ShouldBeNil)
+ })
+
+ Convey("When doing an annotation query with a time column holding a datetime null value should return nil", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT
+ cast(null as datetime) as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `,
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0], ShouldBeNil)
+ })
})
})
}
func InitMSSQLTestDB(t *testing.T) *xorm.Engine {
x, err := xorm.NewEngine(sqlutil.TestDB_Mssql.DriverName, strings.Replace(sqlutil.TestDB_Mssql.ConnStr, "localhost", serverIP, 1))
+ x.DatabaseTZ = time.UTC
+ x.TZLocation = time.UTC
// x.ShowSQL()
@@ -667,8 +859,6 @@ func InitMSSQLTestDB(t *testing.T) *xorm.Engine {
t.Fatalf("Failed to init mssql db %v", err)
}
- sqlutil.CleanDB(x)
-
return x
}
diff --git a/pkg/tsdb/mysql/macros.go b/pkg/tsdb/mysql/macros.go
index b0170070dcf..a292f209429 100644
--- a/pkg/tsdb/mysql/macros.go
+++ b/pkg/tsdb/mysql/macros.go
@@ -68,7 +68,7 @@ func replaceAllStringSubmatchFunc(re *regexp.Regexp, str string, repl func([]str
func (m *MySqlMacroEngine) evaluateMacro(name string, args []string) (string, error) {
switch name {
- case "__time":
+ case "__timeEpoch", "__time":
if len(args) == 0 {
return "", fmt.Errorf("missing time column argument for macro %v", name)
}
diff --git a/pkg/tsdb/mysql/mysql.go b/pkg/tsdb/mysql/mysql.go
index f3060e235e5..483974c55a4 100644
--- a/pkg/tsdb/mysql/mysql.go
+++ b/pkg/tsdb/mysql/mysql.go
@@ -81,7 +81,7 @@ func (e MysqlQueryEndpoint) transformToTable(query *tsdb.Query, rows *core.Rows,
// check if there is a column named time
for i, col := range columnNames {
switch col {
- case "time_sec":
+ case "time", "time_sec":
timeIndex = i
}
}
@@ -96,13 +96,10 @@ func (e MysqlQueryEndpoint) transformToTable(query *tsdb.Query, rows *core.Rows,
return err
}
- // for annotations, convert to epoch
- if timeIndex != -1 {
- switch value := values[timeIndex].(type) {
- case time.Time:
- values[timeIndex] = float64(value.UnixNano() / 1e9)
- }
- }
+ // converts column named time to unix timestamp in milliseconds to make
+ // native mysql datetime types and epoch dates work in
+ // annotation and table queries.
+ tsdb.ConvertSqlTimeColumnToEpochMs(values, timeIndex)
table.Rows = append(table.Rows, values)
}
@@ -185,9 +182,37 @@ func (e MysqlQueryEndpoint) transformToTimeSeries(query *tsdb.Query, rows *core.
return err
}
- rowData := NewStringStringScan(columnNames)
+ columnTypes, err := rows.ColumnTypes()
+ if err != nil {
+ return err
+ }
+
rowLimit := 1000000
rowCount := 0
+ timeIndex := -1
+ metricIndex := -1
+
+ // check columns of resultset: a column named time is mandatory
+ // the first text column is treated as metric name unless a column named metric is present
+ for i, col := range columnNames {
+ switch col {
+ case "time", "time_sec":
+ timeIndex = i
+ case "metric":
+ metricIndex = i
+ default:
+ if metricIndex == -1 {
+ switch columnTypes[i].DatabaseTypeName() {
+ case "CHAR", "VARCHAR", "TINYTEXT", "TEXT", "MEDIUMTEXT", "LONGTEXT":
+ metricIndex = i
+ }
+ }
+ }
+ }
+
+ if timeIndex == -1 {
+ return fmt.Errorf("Found no column named time or time_sec")
+ }
fillMissing := query.Model.Get("fill").MustBool(false)
var fillInterval float64
@@ -198,53 +223,90 @@ func (e MysqlQueryEndpoint) transformToTimeSeries(query *tsdb.Query, rows *core.
fillValue.Float64 = query.Model.Get("fillValue").MustFloat64()
fillValue.Valid = true
}
-
}
- for ; rows.Next(); rowCount++ {
+ for rows.Next() {
+ var timestamp float64
+ var value null.Float
+ var metric string
+
if rowCount > rowLimit {
- return fmt.Errorf("MySQL query row limit exceeded, limit %d", rowLimit)
+ return fmt.Errorf("PostgreSQL query row limit exceeded, limit %d", rowLimit)
}
- err := rowData.Update(rows.Rows)
+ values, err := e.getTypedRowData(rows)
if err != nil {
- e.log.Error("MySQL response parsing", "error", err)
- return fmt.Errorf("MySQL response parsing error %v", err)
+ return err
}
- if rowData.metric == "" {
- rowData.metric = "Unknown"
+ switch columnValue := values[timeIndex].(type) {
+ case int64:
+ timestamp = float64(columnValue * 1000)
+ case float64:
+ timestamp = columnValue * 1000
+ case time.Time:
+ timestamp = float64(columnValue.UnixNano() / 1e6)
+ default:
+ return fmt.Errorf("Invalid type for column time, must be of type timestamp or unix timestamp, got: %T %v", columnValue, columnValue)
}
- if !rowData.time.Valid {
- return fmt.Errorf("Found row with no time value")
- }
-
- series, exist := pointsBySeries[rowData.metric]
- if exist == false {
- series = &tsdb.TimeSeries{Name: rowData.metric}
- pointsBySeries[rowData.metric] = series
- seriesByQueryOrder.PushBack(rowData.metric)
- }
-
- if fillMissing {
- var intervalStart float64
- if exist == false {
- intervalStart = float64(tsdbQuery.TimeRange.MustGetFrom().UnixNano() / 1e6)
+ if metricIndex >= 0 {
+ if columnValue, ok := values[metricIndex].(string); ok == true {
+ metric = columnValue
} else {
- intervalStart = series.Points[len(series.Points)-1][1].Float64 + fillInterval
- }
-
- // align interval start
- intervalStart = math.Floor(intervalStart/fillInterval) * fillInterval
-
- for i := intervalStart; i < rowData.time.Float64; i += fillInterval {
- series.Points = append(series.Points, tsdb.TimePoint{fillValue, null.FloatFrom(i)})
- rowCount++
+ return fmt.Errorf("Column metric must be of type char,varchar or text, got: %T %v", values[metricIndex], values[metricIndex])
}
}
- series.Points = append(series.Points, tsdb.TimePoint{rowData.value, rowData.time})
+ for i, col := range columnNames {
+ if i == timeIndex || i == metricIndex {
+ continue
+ }
+
+ switch columnValue := values[i].(type) {
+ case int64:
+ value = null.FloatFrom(float64(columnValue))
+ case float64:
+ value = null.FloatFrom(columnValue)
+ case nil:
+ value.Valid = false
+ default:
+ return fmt.Errorf("Value column must have numeric datatype, column: %s type: %T value: %v", col, columnValue, columnValue)
+ }
+ if metricIndex == -1 {
+ metric = col
+ }
+
+ series, exist := pointsBySeries[metric]
+ if exist == false {
+ series = &tsdb.TimeSeries{Name: metric}
+ pointsBySeries[metric] = series
+ seriesByQueryOrder.PushBack(metric)
+ }
+
+ if fillMissing {
+ var intervalStart float64
+ if exist == false {
+ intervalStart = float64(tsdbQuery.TimeRange.MustGetFrom().UnixNano() / 1e6)
+ } else {
+ intervalStart = series.Points[len(series.Points)-1][1].Float64 + fillInterval
+ }
+
+ // align interval start
+ intervalStart = math.Floor(intervalStart/fillInterval) * fillInterval
+
+ for i := intervalStart; i < timestamp; i += fillInterval {
+ series.Points = append(series.Points, tsdb.TimePoint{fillValue, null.FloatFrom(i)})
+ rowCount++
+ }
+ }
+
+ series.Points = append(series.Points, tsdb.TimePoint{value, null.FloatFrom(timestamp)})
+
+ e.log.Debug("Rows", "metric", metric, "time", timestamp, "value", value)
+ rowCount++
+
+ }
}
for elem := seriesByQueryOrder.Front(); elem != nil; elem = elem.Next() {
@@ -269,62 +331,3 @@ func (e MysqlQueryEndpoint) transformToTimeSeries(query *tsdb.Query, rows *core.
result.Meta.Set("rowCount", rowCount)
return nil
}
-
-type stringStringScan struct {
- rowPtrs []interface{}
- rowValues []string
- columnNames []string
- columnCount int
-
- time null.Float
- value null.Float
- metric string
-}
-
-func NewStringStringScan(columnNames []string) *stringStringScan {
- s := &stringStringScan{
- columnCount: len(columnNames),
- columnNames: columnNames,
- rowPtrs: make([]interface{}, len(columnNames)),
- rowValues: make([]string, len(columnNames)),
- }
-
- for i := 0; i < s.columnCount; i++ {
- s.rowPtrs[i] = new(sql.RawBytes)
- }
-
- return s
-}
-
-func (s *stringStringScan) Update(rows *sql.Rows) error {
- if err := rows.Scan(s.rowPtrs...); err != nil {
- return err
- }
-
- s.time = null.FloatFromPtr(nil)
- s.value = null.FloatFromPtr(nil)
-
- for i := 0; i < s.columnCount; i++ {
- if rb, ok := s.rowPtrs[i].(*sql.RawBytes); ok {
- s.rowValues[i] = string(*rb)
-
- switch s.columnNames[i] {
- case "time_sec":
- if sec, err := strconv.ParseInt(s.rowValues[i], 10, 64); err == nil {
- s.time = null.FloatFrom(float64(sec * 1000))
- }
- case "value":
- if value, err := strconv.ParseFloat(s.rowValues[i], 64); err == nil {
- s.value = null.FloatFrom(value)
- }
- case "metric":
- s.metric = s.rowValues[i]
- }
-
- *rb = nil // reset pointer to discard current value to avoid a bug
- } else {
- return fmt.Errorf("Cannot convert index %d column %s to type *sql.RawBytes", i, s.columnNames[i])
- }
- }
- return nil
-}
diff --git a/pkg/tsdb/mysql/mysql_test.go b/pkg/tsdb/mysql/mysql_test.go
index fe2c82223d2..750704c9965 100644
--- a/pkg/tsdb/mysql/mysql_test.go
+++ b/pkg/tsdb/mysql/mysql_test.go
@@ -1,6 +1,8 @@
package mysql
import (
+ "fmt"
+ "math/rand"
"testing"
"time"
@@ -14,6 +16,10 @@ import (
// To run this test, remove the Skip from SkipConvey
// and set up a MySQL db named grafana_tests and a user/password grafana/password
+// Use the docker/blocks/mysql_tests/docker-compose.yaml to spin up a
+// preconfigured MySQL server suitable for running these tests.
+// Thers's also a dashboard.json in same directory that you can import to Grafana
+// once you've created a datasource for the test server/database.
func TestMySQL(t *testing.T) {
SkipConvey("MySQL", t, func() {
x := InitMySQLTestDB(t)
@@ -29,110 +35,621 @@ func TestMySQL(t *testing.T) {
sess := x.NewSession()
defer sess.Close()
- sql := "CREATE TABLE `mysql_types` ("
- sql += "`atinyint` tinyint(1) NOT NULL,"
- sql += "`avarchar` varchar(3) NOT NULL,"
- sql += "`achar` char(3),"
- sql += "`amediumint` mediumint NOT NULL,"
- sql += "`asmallint` smallint NOT NULL,"
- sql += "`abigint` bigint NOT NULL,"
- sql += "`aint` int(11) NOT NULL,"
- sql += "`adouble` double(10,2),"
- sql += "`anewdecimal` decimal(10,2),"
- sql += "`afloat` float(10,2) NOT NULL,"
- sql += "`atimestamp` timestamp NOT NULL,"
- sql += "`adatetime` datetime NOT NULL,"
- sql += "`atime` time NOT NULL,"
- // sql += "`ayear` year," // Crashes xorm when running cleandb
- sql += "`abit` bit(1),"
- sql += "`atinytext` tinytext,"
- sql += "`atinyblob` tinyblob,"
- sql += "`atext` text,"
- sql += "`ablob` blob,"
- sql += "`amediumtext` mediumtext,"
- sql += "`amediumblob` mediumblob,"
- sql += "`alongtext` longtext,"
- sql += "`alongblob` longblob,"
- sql += "`aenum` enum('val1', 'val2'),"
- sql += "`aset` set('a', 'b', 'c', 'd'),"
- sql += "`adate` date,"
- sql += "`time_sec` datetime(6),"
- sql += "`aintnull` int(11),"
- sql += "`afloatnull` float(10,2),"
- sql += "`avarcharnull` varchar(3),"
- sql += "`adecimalnull` decimal(10,2)"
- sql += ") ENGINE=InnoDB DEFAULT CHARSET=latin1;"
- _, err := sess.Exec(sql)
- So(err, ShouldBeNil)
+ fromStart := time.Date(2018, 3, 15, 13, 0, 0, 0, time.Local)
- sql = "INSERT INTO `mysql_types` "
- sql += "(`atinyint`, `avarchar`, `achar`, `amediumint`, `asmallint`, `abigint`, `aint`, `adouble`, "
- sql += "`anewdecimal`, `afloat`, `adatetime`, `atimestamp`, `atime`, `abit`, `atinytext`, "
- sql += "`atinyblob`, `atext`, `ablob`, `amediumtext`, `amediumblob`, `alongtext`, `alongblob`, "
- sql += "`aenum`, `aset`, `adate`, `time_sec`) "
- sql += "VALUES(1, 'abc', 'def', 1, 10, 100, 1420070400, 1.11, "
- sql += "2.22, 3.33, now(), current_timestamp(), '11:11:11', 1, 'tinytext', "
- sql += "'tinyblob', 'text', 'blob', 'mediumtext', 'mediumblob', 'longtext', 'longblob', "
- sql += "'val2', 'a,b', curdate(), '2018-01-01 00:01:01.123456');"
- _, err = sess.Exec(sql)
- So(err, ShouldBeNil)
-
- Convey("Query with Table format should map MySQL column types to Go types", func() {
- query := &tsdb.TsdbQuery{
- Queries: []*tsdb.Query{
- {
- Model: simplejson.NewFromAny(map[string]interface{}{
- "rawSql": "SELECT * FROM mysql_types",
- "format": "table",
- }),
- RefId: "A",
- },
- },
+ Convey("Given a table with different native data types", func() {
+ if exists, err := sess.IsTableExist("mysql_types"); err != nil || exists {
+ So(err, ShouldBeNil)
+ sess.DropTable("mysql_types")
}
- resp, err := endpoint.Query(nil, nil, query)
- queryResult := resp.Results["A"]
+ sql := "CREATE TABLE `mysql_types` ("
+ sql += "`atinyint` tinyint(1) NOT NULL,"
+ sql += "`avarchar` varchar(3) NOT NULL,"
+ sql += "`achar` char(3),"
+ sql += "`amediumint` mediumint NOT NULL,"
+ sql += "`asmallint` smallint NOT NULL,"
+ sql += "`abigint` bigint NOT NULL,"
+ sql += "`aint` int(11) NOT NULL,"
+ sql += "`adouble` double(10,2),"
+ sql += "`anewdecimal` decimal(10,2),"
+ sql += "`afloat` float(10,2) NOT NULL,"
+ sql += "`atimestamp` timestamp NOT NULL,"
+ sql += "`adatetime` datetime NOT NULL,"
+ sql += "`atime` time NOT NULL,"
+ sql += "`ayear` year," // Crashes xorm when running cleandb
+ sql += "`abit` bit(1),"
+ sql += "`atinytext` tinytext,"
+ sql += "`atinyblob` tinyblob,"
+ sql += "`atext` text,"
+ sql += "`ablob` blob,"
+ sql += "`amediumtext` mediumtext,"
+ sql += "`amediumblob` mediumblob,"
+ sql += "`alongtext` longtext,"
+ sql += "`alongblob` longblob,"
+ sql += "`aenum` enum('val1', 'val2'),"
+ sql += "`aset` set('a', 'b', 'c', 'd'),"
+ sql += "`adate` date,"
+ sql += "`time_sec` datetime(6),"
+ sql += "`aintnull` int(11),"
+ sql += "`afloatnull` float(10,2),"
+ sql += "`avarcharnull` varchar(3),"
+ sql += "`adecimalnull` decimal(10,2)"
+ sql += ") ENGINE=InnoDB DEFAULT CHARSET=latin1;"
+ _, err := sess.Exec(sql)
So(err, ShouldBeNil)
- column := queryResult.Tables[0].Rows[0]
+ sql = "INSERT INTO `mysql_types` "
+ sql += "(`atinyint`, `avarchar`, `achar`, `amediumint`, `asmallint`, `abigint`, `aint`, `adouble`, "
+ sql += "`anewdecimal`, `afloat`, `adatetime`, `atimestamp`, `atime`, `ayear`, `abit`, `atinytext`, "
+ sql += "`atinyblob`, `atext`, `ablob`, `amediumtext`, `amediumblob`, `alongtext`, `alongblob`, "
+ sql += "`aenum`, `aset`, `adate`, `time_sec`) "
+ sql += "VALUES(1, 'abc', 'def', 1, 10, 100, 1420070400, 1.11, "
+ sql += "2.22, 3.33, now(), current_timestamp(), '11:11:11', '2018', 1, 'tinytext', "
+ sql += "'tinyblob', 'text', 'blob', 'mediumtext', 'mediumblob', 'longtext', 'longblob', "
+ sql += "'val2', 'a,b', curdate(), '2018-01-01 00:01:01.123456');"
+ _, err = sess.Exec(sql)
+ So(err, ShouldBeNil)
- So(*column[0].(*int8), ShouldEqual, 1)
- So(column[1].(string), ShouldEqual, "abc")
- So(column[2].(string), ShouldEqual, "def")
- So(*column[3].(*int32), ShouldEqual, 1)
- So(*column[4].(*int16), ShouldEqual, 10)
- So(*column[5].(*int64), ShouldEqual, 100)
- So(*column[6].(*int32), ShouldEqual, 1420070400)
- So(column[7].(float64), ShouldEqual, 1.11)
- So(column[8].(float64), ShouldEqual, 2.22)
- So(*column[9].(*float32), ShouldEqual, 3.33)
- _, offset := time.Now().Zone()
- So(column[10].(time.Time), ShouldHappenWithin, time.Duration(10*time.Second), time.Now().Add(time.Duration(offset)*time.Second))
- So(column[11].(time.Time), ShouldHappenWithin, time.Duration(10*time.Second), time.Now().Add(time.Duration(offset)*time.Second))
- So(column[12].(string), ShouldEqual, "11:11:11")
- So(*column[13].(*[]byte), ShouldHaveSameTypeAs, []byte{1})
- So(column[14].(string), ShouldEqual, "tinytext")
- So(column[15].(string), ShouldEqual, "tinyblob")
- So(column[16].(string), ShouldEqual, "text")
- So(column[17].(string), ShouldEqual, "blob")
- So(column[18].(string), ShouldEqual, "mediumtext")
- So(column[19].(string), ShouldEqual, "mediumblob")
- So(column[20].(string), ShouldEqual, "longtext")
- So(column[21].(string), ShouldEqual, "longblob")
- So(column[22].(string), ShouldEqual, "val2")
- So(column[23].(string), ShouldEqual, "a,b")
- So(column[24].(time.Time).Format("2006-01-02T00:00:00Z"), ShouldEqual, time.Now().Format("2006-01-02T00:00:00Z"))
- So(column[25].(float64), ShouldEqual, 1514764861)
- So(column[26], ShouldEqual, nil)
- So(column[27], ShouldEqual, nil)
- So(column[28], ShouldEqual, "")
- So(column[29], ShouldEqual, nil)
+ Convey("Query with Table format should map MySQL column types to Go types", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": "SELECT * FROM mysql_types",
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ column := queryResult.Tables[0].Rows[0]
+
+ So(*column[0].(*int8), ShouldEqual, 1)
+ So(column[1].(string), ShouldEqual, "abc")
+ So(column[2].(string), ShouldEqual, "def")
+ So(*column[3].(*int32), ShouldEqual, 1)
+ So(*column[4].(*int16), ShouldEqual, 10)
+ So(*column[5].(*int64), ShouldEqual, 100)
+ So(*column[6].(*int32), ShouldEqual, 1420070400)
+ So(column[7].(float64), ShouldEqual, 1.11)
+ So(column[8].(float64), ShouldEqual, 2.22)
+ So(*column[9].(*float32), ShouldEqual, 3.33)
+ _, offset := time.Now().Zone()
+ So(column[10].(time.Time), ShouldHappenWithin, time.Duration(10*time.Second), time.Now().Add(time.Duration(offset)*time.Second))
+ So(column[11].(time.Time), ShouldHappenWithin, time.Duration(10*time.Second), time.Now().Add(time.Duration(offset)*time.Second))
+ So(column[12].(string), ShouldEqual, "11:11:11")
+ So(column[13].(int64), ShouldEqual, 2018)
+ So(*column[14].(*[]byte), ShouldHaveSameTypeAs, []byte{1})
+ So(column[15].(string), ShouldEqual, "tinytext")
+ So(column[16].(string), ShouldEqual, "tinyblob")
+ So(column[17].(string), ShouldEqual, "text")
+ So(column[18].(string), ShouldEqual, "blob")
+ So(column[19].(string), ShouldEqual, "mediumtext")
+ So(column[20].(string), ShouldEqual, "mediumblob")
+ So(column[21].(string), ShouldEqual, "longtext")
+ So(column[22].(string), ShouldEqual, "longblob")
+ So(column[23].(string), ShouldEqual, "val2")
+ So(column[24].(string), ShouldEqual, "a,b")
+ So(column[25].(time.Time).Format("2006-01-02T00:00:00Z"), ShouldEqual, time.Now().Format("2006-01-02T00:00:00Z"))
+ So(column[26].(float64), ShouldEqual, float64(1514764861000))
+ So(column[27], ShouldEqual, nil)
+ So(column[28], ShouldEqual, nil)
+ So(column[29], ShouldEqual, "")
+ So(column[30], ShouldEqual, nil)
+ })
+ })
+
+ Convey("Given a table with metrics that lacks data for some series ", func() {
+ type metric struct {
+ Time time.Time
+ Value int64
+ }
+
+ if exist, err := sess.IsTableExist(metric{}); err != nil || exist {
+ So(err, ShouldBeNil)
+ sess.DropTable(metric{})
+ }
+ err := sess.CreateTable(metric{})
+ So(err, ShouldBeNil)
+
+ series := []*metric{}
+ firstRange := genTimeRangeByInterval(fromStart, 10*time.Minute, 10*time.Second)
+ secondRange := genTimeRangeByInterval(fromStart.Add(20*time.Minute), 10*time.Minute, 10*time.Second)
+
+ for _, t := range firstRange {
+ series = append(series, &metric{
+ Time: t,
+ Value: 15,
+ })
+ }
+
+ for _, t := range secondRange {
+ series = append(series, &metric{
+ Time: t,
+ Value: 20,
+ })
+ }
+
+ for _, s := range series {
+ _, err = sess.Insert(s)
+ So(err, ShouldBeNil)
+ }
+
+ Convey("When doing a metric query using timeGroup", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": "SELECT $__timeGroup(time, '5m') as time_sec, avg(value) as value FROM metric GROUP BY 1 ORDER BY 1",
+ "format": "time_series",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ points := queryResult.Series[0].Points
+ So(len(points), ShouldEqual, 6)
+
+ dt := fromStart
+
+ for i := 0; i < 3; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 15)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
+
+ // adjust for 5 minute gap
+ dt = dt.Add(5 * time.Minute)
+ for i := 3; i < 6; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 20)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
+ })
+
+ Convey("When doing a metric query using timeGroup with NULL fill enabled", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": "SELECT $__timeGroup(time, '5m', NULL) as time_sec, avg(value) as value FROM metric GROUP BY 1 ORDER BY 1",
+ "format": "time_series",
+ }),
+ RefId: "A",
+ },
+ },
+ TimeRange: &tsdb.TimeRange{
+ From: fmt.Sprintf("%v", fromStart.Unix()*1000),
+ To: fmt.Sprintf("%v", fromStart.Add(34*time.Minute).Unix()*1000),
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ points := queryResult.Series[0].Points
+ So(len(points), ShouldEqual, 7)
+
+ dt := fromStart
+
+ for i := 0; i < 3; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 15)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
+
+ So(points[3][0].Valid, ShouldBeFalse)
+
+ // adjust for 5 minute gap
+ dt = dt.Add(5 * time.Minute)
+ for i := 4; i < 7; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 20)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
+ })
+
+ Convey("When doing a metric query using timeGroup with float fill enabled", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": "SELECT $__timeGroup(time, '5m', 1.5) as time_sec, avg(value) as value FROM metric GROUP BY 1 ORDER BY 1",
+ "format": "time_series",
+ }),
+ RefId: "A",
+ },
+ },
+ TimeRange: &tsdb.TimeRange{
+ From: fmt.Sprintf("%v", fromStart.Unix()*1000),
+ To: fmt.Sprintf("%v", fromStart.Add(34*time.Minute).Unix()*1000),
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ points := queryResult.Series[0].Points
+ So(points[3][0].Float64, ShouldEqual, 1.5)
+ })
+ })
+
+ Convey("Given a table with metrics having multiple values and measurements", func() {
+ type metric_values struct {
+ Time time.Time
+ Measurement string
+ ValueOne int64 `xorm:"integer 'valueOne'"`
+ ValueTwo int64 `xorm:"integer 'valueTwo'"`
+ }
+
+ if exist, err := sess.IsTableExist(metric_values{}); err != nil || exist {
+ So(err, ShouldBeNil)
+ sess.DropTable(metric_values{})
+ }
+ err := sess.CreateTable(metric_values{})
+ So(err, ShouldBeNil)
+
+ rand.Seed(time.Now().Unix())
+ rnd := func(min, max int64) int64 {
+ return rand.Int63n(max-min) + min
+ }
+
+ series := []*metric_values{}
+ for _, t := range genTimeRangeByInterval(fromStart.Add(-30*time.Minute), 90*time.Minute, 5*time.Minute) {
+ series = append(series, &metric_values{
+ Time: t,
+ Measurement: "Metric A",
+ ValueOne: rnd(0, 100),
+ ValueTwo: rnd(0, 100),
+ })
+ series = append(series, &metric_values{
+ Time: t,
+ Measurement: "Metric B",
+ ValueOne: rnd(0, 100),
+ ValueTwo: rnd(0, 100),
+ })
+ }
+
+ for _, s := range series {
+ _, err := sess.Insert(s)
+ So(err, ShouldBeNil)
+ }
+
+ Convey("When doing a metric query grouping by time and select metric column should return correct series", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT $__time(time), CONCAT(measurement, ' - value one') as metric, valueOne FROM metric_values ORDER BY 1`,
+ "format": "time_series",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ So(len(queryResult.Series), ShouldEqual, 2)
+ So(queryResult.Series[0].Name, ShouldEqual, "Metric B - value one")
+ So(queryResult.Series[1].Name, ShouldEqual, "Metric A - value one")
+ })
+
+ Convey("When doing a metric query grouping by time should return correct series", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT $__time(time), valueOne, valueTwo FROM metric_values ORDER BY 1`,
+ "format": "time_series",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ So(len(queryResult.Series), ShouldEqual, 2)
+ So(queryResult.Series[0].Name, ShouldEqual, "valueOne")
+ So(queryResult.Series[1].Name, ShouldEqual, "valueTwo")
+ })
+ })
+
+ Convey("Given a table with event data", func() {
+ type event struct {
+ TimeSec int64
+ Description string
+ Tags string
+ }
+
+ if exist, err := sess.IsTableExist(event{}); err != nil || exist {
+ So(err, ShouldBeNil)
+ sess.DropTable(event{})
+ }
+ err := sess.CreateTable(event{})
+ So(err, ShouldBeNil)
+
+ events := []*event{}
+ for _, t := range genTimeRangeByInterval(fromStart.Add(-20*time.Minute), 60*time.Minute, 25*time.Minute) {
+ events = append(events, &event{
+ TimeSec: t.Unix(),
+ Description: "Someone deployed something",
+ Tags: "deploy",
+ })
+ events = append(events, &event{
+ TimeSec: t.Add(5 * time.Minute).Unix(),
+ Description: "New support ticket registered",
+ Tags: "ticket",
+ })
+ }
+
+ for _, e := range events {
+ _, err = sess.Insert(e)
+ So(err, ShouldBeNil)
+ }
+
+ Convey("When doing an annotation query of deploy events should return expected result", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT time_sec, description as text, tags FROM event WHERE $__unixEpochFilter(time_sec) AND tags='deploy' ORDER BY 1 ASC`,
+ "format": "table",
+ }),
+ RefId: "Deploys",
+ },
+ },
+ TimeRange: &tsdb.TimeRange{
+ From: fmt.Sprintf("%v", fromStart.Add(-20*time.Minute).Unix()*1000),
+ To: fmt.Sprintf("%v", fromStart.Add(40*time.Minute).Unix()*1000),
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ queryResult := resp.Results["Deploys"]
+ So(err, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 3)
+ })
+
+ Convey("When doing an annotation query of ticket events should return expected result", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT time_sec, description as text, tags FROM event WHERE $__unixEpochFilter(time_sec) AND tags='ticket' ORDER BY 1 ASC`,
+ "format": "table",
+ }),
+ RefId: "Tickets",
+ },
+ },
+ TimeRange: &tsdb.TimeRange{
+ From: fmt.Sprintf("%v", fromStart.Add(-20*time.Minute).Unix()*1000),
+ To: fmt.Sprintf("%v", fromStart.Add(40*time.Minute).Unix()*1000),
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ queryResult := resp.Results["Tickets"]
+ So(err, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 3)
+ })
+
+ Convey("When doing an annotation query with a time column in datetime format", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 0, time.UTC)
+ dtFormat := "2006-01-02 15:04:05.999999999"
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ CAST('%s' as datetime) as time_sec,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Format(dtFormat)),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(float64), ShouldEqual, float64(dt.Unix()*1000))
+ })
+
+ Convey("When doing an annotation query with a time column in epoch second format should return ms", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 527e6, time.UTC)
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ %d as time_sec,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Unix()),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(int64), ShouldEqual, dt.Unix()*1000)
+ })
+
+ Convey("When doing an annotation query with a time column in epoch second format (signed integer) should return ms", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 0, time.Local)
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ CAST('%d' as signed integer) as time_sec,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Unix()),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(int64), ShouldEqual, int64(dt.Unix()*1000))
+ })
+
+ Convey("When doing an annotation query with a time column in epoch millisecond format should return ms", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 527e6, time.UTC)
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ %d as time_sec,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Unix()*1000),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(int64), ShouldEqual, dt.Unix()*1000)
+ })
+
+ Convey("When doing an annotation query with a time column holding a unsigned integer null value should return nil", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT
+ cast(null as unsigned integer) as time_sec,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `,
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0], ShouldBeNil)
+ })
+
+ Convey("When doing an annotation query with a time column holding a DATETIME null value should return nil", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT
+ cast(null as DATETIME) as time_sec,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `,
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0], ShouldBeNil)
+ })
})
})
}
func InitMySQLTestDB(t *testing.T) *xorm.Engine {
x, err := xorm.NewEngine(sqlutil.TestDB_Mysql.DriverName, sqlutil.TestDB_Mysql.ConnStr+"&parseTime=true")
+ x.DatabaseTZ = time.Local
+ x.TZLocation = time.Local
// x.ShowSQL()
@@ -140,7 +657,18 @@ func InitMySQLTestDB(t *testing.T) *xorm.Engine {
t.Fatalf("Failed to init mysql db %v", err)
}
- sqlutil.CleanDB(x)
-
return x
}
+
+func genTimeRangeByInterval(from time.Time, duration time.Duration, interval time.Duration) []time.Time {
+ durationSec := int64(duration.Seconds())
+ intervalSec := int64(interval.Seconds())
+ timeRange := []time.Time{}
+
+ for i := int64(0); i < durationSec; i += intervalSec {
+ timeRange = append(timeRange, from)
+ from = from.Add(time.Duration(int64(time.Second) * intervalSec))
+ }
+
+ return timeRange
+}
diff --git a/pkg/tsdb/postgres/postgres.go b/pkg/tsdb/postgres/postgres.go
index 6a084ad1237..5f6b56ebcf1 100644
--- a/pkg/tsdb/postgres/postgres.go
+++ b/pkg/tsdb/postgres/postgres.go
@@ -63,7 +63,6 @@ func (e *PostgresQueryEndpoint) Query(ctx context.Context, dsInfo *models.DataSo
}
func (e PostgresQueryEndpoint) transformToTable(query *tsdb.Query, rows *core.Rows, result *tsdb.QueryResult, tsdbQuery *tsdb.TsdbQuery) error {
-
columnNames, err := rows.Columns()
if err != nil {
return err
@@ -100,14 +99,10 @@ func (e PostgresQueryEndpoint) transformToTable(query *tsdb.Query, rows *core.Ro
return err
}
- // convert column named time to unix timestamp to make
- // native datetime postgres types work in annotation queries
- if timeIndex != -1 {
- switch value := values[timeIndex].(type) {
- case time.Time:
- values[timeIndex] = float64(value.UnixNano() / 1e9)
- }
- }
+ // converts column named time to unix timestamp in milliseconds to make
+ // native postgres datetime types and epoch dates work in
+ // annotation and table queries.
+ tsdb.ConvertSqlTimeColumnToEpochMs(values, timeIndex)
table.Rows = append(table.Rows, values)
}
@@ -118,7 +113,6 @@ func (e PostgresQueryEndpoint) transformToTable(query *tsdb.Query, rows *core.Ro
}
func (e PostgresQueryEndpoint) getTypedRowData(rows *core.Rows) (tsdb.RowValues, error) {
-
types, err := rows.ColumnTypes()
if err != nil {
return nil, err
@@ -209,7 +203,6 @@ func (e PostgresQueryEndpoint) transformToTimeSeries(query *tsdb.Query, rows *co
fillValue.Float64 = query.Model.Get("fillValue").MustFloat64()
fillValue.Valid = true
}
-
}
for rows.Next() {
diff --git a/pkg/tsdb/postgres/postgres_test.go b/pkg/tsdb/postgres/postgres_test.go
index 75e8cb77f2e..3f2203ac7a4 100644
--- a/pkg/tsdb/postgres/postgres_test.go
+++ b/pkg/tsdb/postgres/postgres_test.go
@@ -1,6 +1,8 @@
package postgres
import (
+ "fmt"
+ "math/rand"
"testing"
"time"
@@ -14,7 +16,11 @@ import (
)
// To run this test, remove the Skip from SkipConvey
-// and set up a PostgreSQL db named grafanatest and a user/password grafanatest/grafanatest
+// and set up a PostgreSQL db named grafanatest and a user/password grafanatest/grafanatest!
+// Use the docker/blocks/postgres_tests/docker-compose.yaml to spin up a
+// preconfigured Postgres server suitable for running these tests.
+// Thers's also a dashboard.json in same directory that you can import to Grafana
+// once you've created a datasource for the test server/database.
func TestPostgres(t *testing.T) {
SkipConvey("PostgreSQL", t, func() {
x := InitPostgresTestDB(t)
@@ -30,88 +36,599 @@ func TestPostgres(t *testing.T) {
sess := x.NewSession()
defer sess.Close()
- sql := `
- CREATE TABLE postgres_types(
- c00_smallint smallint,
- c01_integer integer,
- c02_bigint bigint,
+ fromStart := time.Date(2018, 3, 15, 13, 0, 0, 0, time.UTC).In(time.Local)
- c03_real real,
- c04_double double precision,
- c05_decimal decimal(10,2),
- c06_numeric numeric(10,2),
+ Convey("Given a table with different native data types", func() {
+ sql := `
+ DROP TABLE IF EXISTS postgres_types;
+ CREATE TABLE postgres_types(
+ c00_smallint smallint,
+ c01_integer integer,
+ c02_bigint bigint,
- c07_char char(10),
- c08_varchar varchar(10),
- c09_text text,
+ c03_real real,
+ c04_double double precision,
+ c05_decimal decimal(10,2),
+ c06_numeric numeric(10,2),
- c10_timestamp timestamp without time zone,
- c11_timestamptz timestamp with time zone,
- c12_date date,
- c13_time time without time zone,
- c14_timetz time with time zone,
- c15_interval interval
- );
- `
- _, err := sess.Exec(sql)
- So(err, ShouldBeNil)
+ c07_char char(10),
+ c08_varchar varchar(10),
+ c09_text text,
- sql = `
- INSERT INTO postgres_types VALUES(
- 1,2,3,
- 4.5,6.7,1.1,1.2,
- 'char10','varchar10','text',
+ c10_timestamp timestamp without time zone,
+ c11_timestamptz timestamp with time zone,
+ c12_date date,
+ c13_time time without time zone,
+ c14_timetz time with time zone,
- now(),now(),now(),now(),now(),'15m'::interval
- );
- `
- _, err = sess.Exec(sql)
- So(err, ShouldBeNil)
-
- Convey("Query with Table format should map PostgreSQL column types to Go types", func() {
- query := &tsdb.TsdbQuery{
- Queries: []*tsdb.Query{
- {
- Model: simplejson.NewFromAny(map[string]interface{}{
- "rawSql": "SELECT * FROM postgres_types",
- "format": "table",
- }),
- RefId: "A",
- },
- },
- }
-
- resp, err := endpoint.Query(nil, nil, query)
- queryResult := resp.Results["A"]
+ c15_interval interval
+ );
+ `
+ _, err := sess.Exec(sql)
So(err, ShouldBeNil)
- column := queryResult.Tables[0].Rows[0]
- So(column[0].(int64), ShouldEqual, 1)
- So(column[1].(int64), ShouldEqual, 2)
- So(column[2].(int64), ShouldEqual, 3)
- So(column[3].(float64), ShouldEqual, 4.5)
- So(column[4].(float64), ShouldEqual, 6.7)
- // libpq doesnt properly convert decimal, numeric and char to go types but returns []uint8 instead
- // So(column[5].(float64), ShouldEqual, 1.1)
- // So(column[6].(float64), ShouldEqual, 1.2)
- // So(column[7].(string), ShouldEqual, "char")
- So(column[8].(string), ShouldEqual, "varchar10")
- So(column[9].(string), ShouldEqual, "text")
+ sql = `
+ INSERT INTO postgres_types VALUES(
+ 1,2,3,
+ 4.5,6.7,1.1,1.2,
+ 'char10','varchar10','text',
- So(column[10].(time.Time), ShouldHaveSameTypeAs, time.Now())
- So(column[11].(time.Time), ShouldHaveSameTypeAs, time.Now())
- So(column[12].(time.Time), ShouldHaveSameTypeAs, time.Now())
- So(column[13].(time.Time), ShouldHaveSameTypeAs, time.Now())
- So(column[14].(time.Time), ShouldHaveSameTypeAs, time.Now())
+ now(),now(),now(),now(),now(),'15m'::interval
+ );
+ `
+ _, err = sess.Exec(sql)
+ So(err, ShouldBeNil)
- // libpq doesnt properly convert interval to go types but returns []uint8 instead
- // So(column[15].(time.Time), ShouldHaveSameTypeAs, time.Now())
+ Convey("When doing a table query should map Postgres column types to Go types", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": "SELECT * FROM postgres_types",
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ column := queryResult.Tables[0].Rows[0]
+ So(column[0].(int64), ShouldEqual, 1)
+ So(column[1].(int64), ShouldEqual, 2)
+ So(column[2].(int64), ShouldEqual, 3)
+
+ So(column[3].(float64), ShouldEqual, 4.5)
+ So(column[4].(float64), ShouldEqual, 6.7)
+ So(column[5].(float64), ShouldEqual, 1.1)
+ So(column[6].(float64), ShouldEqual, 1.2)
+
+ So(column[7].(string), ShouldEqual, "char10 ")
+ So(column[8].(string), ShouldEqual, "varchar10")
+ So(column[9].(string), ShouldEqual, "text")
+
+ So(column[10].(time.Time), ShouldHaveSameTypeAs, time.Now())
+ So(column[11].(time.Time), ShouldHaveSameTypeAs, time.Now())
+ So(column[12].(time.Time), ShouldHaveSameTypeAs, time.Now())
+ So(column[13].(time.Time), ShouldHaveSameTypeAs, time.Now())
+ So(column[14].(time.Time), ShouldHaveSameTypeAs, time.Now())
+
+ So(column[15].(string), ShouldEqual, "00:15:00")
+ })
+ })
+
+ Convey("Given a table with metrics that lacks data for some series ", func() {
+ sql := `
+ DROP TABLE IF EXISTS metric;
+ CREATE TABLE metric (
+ time timestamp,
+ value integer
+ )
+ `
+
+ _, err := sess.Exec(sql)
+ So(err, ShouldBeNil)
+
+ type metric struct {
+ Time time.Time
+ Value int64
+ }
+
+ series := []*metric{}
+ firstRange := genTimeRangeByInterval(fromStart, 10*time.Minute, 10*time.Second)
+ secondRange := genTimeRangeByInterval(fromStart.Add(20*time.Minute), 10*time.Minute, 10*time.Second)
+
+ for _, t := range firstRange {
+ series = append(series, &metric{
+ Time: t,
+ Value: 15,
+ })
+ }
+
+ for _, t := range secondRange {
+ series = append(series, &metric{
+ Time: t,
+ Value: 20,
+ })
+ }
+
+ for _, s := range series {
+ _, err = sess.Insert(s)
+ So(err, ShouldBeNil)
+ }
+
+ Convey("When doing a metric query using timeGroup", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": "SELECT $__timeGroup(time, '5m'), avg(value) as value FROM metric GROUP BY 1 ORDER BY 1",
+ "format": "time_series",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ points := queryResult.Series[0].Points
+ So(len(points), ShouldEqual, 6)
+
+ dt := fromStart
+
+ for i := 0; i < 3; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 15)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
+
+ // adjust for 5 minute gap
+ dt = dt.Add(5 * time.Minute)
+ for i := 3; i < 6; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 20)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
+ })
+
+ Convey("When doing a metric query using timeGroup with NULL fill enabled", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": "SELECT $__timeGroup(time, '5m', NULL), avg(value) as value FROM metric GROUP BY 1 ORDER BY 1",
+ "format": "time_series",
+ }),
+ RefId: "A",
+ },
+ },
+ TimeRange: &tsdb.TimeRange{
+ From: fmt.Sprintf("%v", fromStart.Unix()*1000),
+ To: fmt.Sprintf("%v", fromStart.Add(34*time.Minute).Unix()*1000),
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ points := queryResult.Series[0].Points
+ So(len(points), ShouldEqual, 7)
+
+ dt := fromStart
+
+ for i := 0; i < 3; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 15)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
+
+ So(points[3][0].Valid, ShouldBeFalse)
+
+ // adjust for 5 minute gap
+ dt = dt.Add(5 * time.Minute)
+ for i := 4; i < 7; i++ {
+ aValue := points[i][0].Float64
+ aTime := time.Unix(int64(points[i][1].Float64)/1000, 0)
+ So(aValue, ShouldEqual, 20)
+ So(aTime, ShouldEqual, dt)
+ dt = dt.Add(5 * time.Minute)
+ }
+ })
+
+ Convey("When doing a metric query using timeGroup with float fill enabled", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": "SELECT $__timeGroup(time, '5m', 1.5), avg(value) as value FROM metric GROUP BY 1 ORDER BY 1",
+ "format": "time_series",
+ }),
+ RefId: "A",
+ },
+ },
+ TimeRange: &tsdb.TimeRange{
+ From: fmt.Sprintf("%v", fromStart.Unix()*1000),
+ To: fmt.Sprintf("%v", fromStart.Add(34*time.Minute).Unix()*1000),
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ points := queryResult.Series[0].Points
+ So(points[3][0].Float64, ShouldEqual, 1.5)
+ })
+ })
+
+ Convey("Given a table with metrics having multiple values and measurements", func() {
+ type metric_values struct {
+ Time time.Time
+ Measurement string
+ ValueOne int64 `xorm:"integer 'valueOne'"`
+ ValueTwo int64 `xorm:"integer 'valueTwo'"`
+ }
+
+ if exist, err := sess.IsTableExist(metric_values{}); err != nil || exist {
+ So(err, ShouldBeNil)
+ sess.DropTable(metric_values{})
+ }
+ err := sess.CreateTable(metric_values{})
+ So(err, ShouldBeNil)
+
+ rand.Seed(time.Now().Unix())
+ rnd := func(min, max int64) int64 {
+ return rand.Int63n(max-min) + min
+ }
+
+ series := []*metric_values{}
+ for _, t := range genTimeRangeByInterval(fromStart.Add(-30*time.Minute), 90*time.Minute, 5*time.Minute) {
+ series = append(series, &metric_values{
+ Time: t,
+ Measurement: "Metric A",
+ ValueOne: rnd(0, 100),
+ ValueTwo: rnd(0, 100),
+ })
+ series = append(series, &metric_values{
+ Time: t,
+ Measurement: "Metric B",
+ ValueOne: rnd(0, 100),
+ ValueTwo: rnd(0, 100),
+ })
+ }
+
+ for _, s := range series {
+ _, err := sess.Insert(s)
+ So(err, ShouldBeNil)
+ }
+
+ Convey("When doing a metric query grouping by time and select metric column should return correct series", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT $__timeEpoch(time), measurement || ' - value one' as metric, "valueOne" FROM metric_values ORDER BY 1`,
+ "format": "time_series",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ So(len(queryResult.Series), ShouldEqual, 2)
+ So(queryResult.Series[0].Name, ShouldEqual, "Metric A - value one")
+ So(queryResult.Series[1].Name, ShouldEqual, "Metric B - value one")
+ })
+
+ Convey("When doing a metric query grouping by time should return correct series", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT $__timeEpoch(time), "valueOne", "valueTwo" FROM metric_values ORDER BY 1`,
+ "format": "time_series",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+
+ So(len(queryResult.Series), ShouldEqual, 2)
+ So(queryResult.Series[0].Name, ShouldEqual, "valueOne")
+ So(queryResult.Series[1].Name, ShouldEqual, "valueTwo")
+ })
+ })
+
+ Convey("Given a table with event data", func() {
+ type event struct {
+ TimeSec int64
+ Description string
+ Tags string
+ }
+
+ if exist, err := sess.IsTableExist(event{}); err != nil || exist {
+ So(err, ShouldBeNil)
+ sess.DropTable(event{})
+ }
+ err := sess.CreateTable(event{})
+ So(err, ShouldBeNil)
+
+ events := []*event{}
+ for _, t := range genTimeRangeByInterval(fromStart.Add(-20*time.Minute), 60*time.Minute, 25*time.Minute) {
+ events = append(events, &event{
+ TimeSec: t.Unix(),
+ Description: "Someone deployed something",
+ Tags: "deploy",
+ })
+ events = append(events, &event{
+ TimeSec: t.Add(5 * time.Minute).Unix(),
+ Description: "New support ticket registered",
+ Tags: "ticket",
+ })
+ }
+
+ for _, e := range events {
+ _, err = sess.Insert(e)
+ So(err, ShouldBeNil)
+ }
+
+ Convey("When doing an annotation query of deploy events should return expected result", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT "time_sec" as time, description as text, tags FROM event WHERE $__unixEpochFilter(time_sec) AND tags='deploy' ORDER BY 1 ASC`,
+ "format": "table",
+ }),
+ RefId: "Deploys",
+ },
+ },
+ TimeRange: &tsdb.TimeRange{
+ From: fmt.Sprintf("%v", fromStart.Add(-20*time.Minute).Unix()*1000),
+ To: fmt.Sprintf("%v", fromStart.Add(40*time.Minute).Unix()*1000),
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ queryResult := resp.Results["Deploys"]
+ So(err, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 3)
+ })
+
+ Convey("When doing an annotation query of ticket events should return expected result", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT "time_sec" as time, description as text, tags FROM event WHERE $__unixEpochFilter(time_sec) AND tags='ticket' ORDER BY 1 ASC`,
+ "format": "table",
+ }),
+ RefId: "Tickets",
+ },
+ },
+ TimeRange: &tsdb.TimeRange{
+ From: fmt.Sprintf("%v", fromStart.Add(-20*time.Minute).Unix()*1000),
+ To: fmt.Sprintf("%v", fromStart.Add(40*time.Minute).Unix()*1000),
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ queryResult := resp.Results["Tickets"]
+ So(err, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 3)
+ })
+
+ Convey("When doing an annotation query with a time column in datetime format", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 527e6, time.UTC)
+ dtFormat := "2006-01-02 15:04:05.999999999"
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ CAST('%s' AS TIMESTAMP) as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Format(dtFormat)),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(float64), ShouldEqual, float64(dt.Unix()*1000))
+ })
+
+ Convey("When doing an annotation query with a time column in epoch second format should return ms", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 527e6, time.UTC)
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ %d as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Unix()),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(int64), ShouldEqual, int64(dt.Unix()*1000))
+ })
+
+ Convey("When doing an annotation query with a time column in epoch second format (int) should return ms", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 527e6, time.UTC)
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ cast(%d as bigint) as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Unix()),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(int64), ShouldEqual, int64(dt.Unix()*1000))
+ })
+
+ Convey("When doing an annotation query with a time column in epoch millisecond format should return ms", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 527e6, time.UTC)
+
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": fmt.Sprintf(`SELECT
+ %d as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `, dt.Unix()*1000),
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0].(int64), ShouldEqual, dt.Unix()*1000)
+ })
+
+ Convey("When doing an annotation query with a time column holding a bigint null value should return nil", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT
+ cast(null as bigint) as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `,
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0], ShouldBeNil)
+ })
+
+ Convey("When doing an annotation query with a time column holding a timestamp null value should return nil", func() {
+ query := &tsdb.TsdbQuery{
+ Queries: []*tsdb.Query{
+ {
+ Model: simplejson.NewFromAny(map[string]interface{}{
+ "rawSql": `SELECT
+ cast(null as timestamp) as time,
+ 'message' as text,
+ 'tag1,tag2' as tags
+ `,
+ "format": "table",
+ }),
+ RefId: "A",
+ },
+ },
+ }
+
+ resp, err := endpoint.Query(nil, nil, query)
+ So(err, ShouldBeNil)
+ queryResult := resp.Results["A"]
+ So(queryResult.Error, ShouldBeNil)
+ So(len(queryResult.Tables[0].Rows), ShouldEqual, 1)
+ columns := queryResult.Tables[0].Rows[0]
+
+ //Should be in milliseconds
+ So(columns[0], ShouldBeNil)
+ })
})
})
}
func InitPostgresTestDB(t *testing.T) *xorm.Engine {
x, err := xorm.NewEngine(sqlutil.TestDB_Postgres.DriverName, sqlutil.TestDB_Postgres.ConnStr)
+ x.DatabaseTZ = time.UTC
+ x.TZLocation = time.UTC
// x.ShowSQL()
@@ -119,7 +636,18 @@ func InitPostgresTestDB(t *testing.T) *xorm.Engine {
t.Fatalf("Failed to init postgres db %v", err)
}
- sqlutil.CleanDB(x)
-
return x
}
+
+func genTimeRangeByInterval(from time.Time, duration time.Duration, interval time.Duration) []time.Time {
+ durationSec := int64(duration.Seconds())
+ intervalSec := int64(interval.Seconds())
+ timeRange := []time.Time{}
+
+ for i := int64(0); i < durationSec; i += intervalSec {
+ timeRange = append(timeRange, from)
+ from = from.Add(time.Duration(int64(time.Second) * intervalSec))
+ }
+
+ return timeRange
+}
diff --git a/pkg/tsdb/sql_engine.go b/pkg/tsdb/sql_engine.go
index 7ea0682235f..16370a4ea7f 100644
--- a/pkg/tsdb/sql_engine.go
+++ b/pkg/tsdb/sql_engine.go
@@ -3,6 +3,7 @@ package tsdb
import (
"context"
"sync"
+ "time"
"github.com/go-xorm/core"
"github.com/go-xorm/xorm"
@@ -133,3 +134,30 @@ func (e *DefaultSqlEngine) Query(
return result, nil
}
+
+// ConvertTimeColumnToEpochMs converts column named time to unix timestamp in milliseconds
+// to make native datetime types and epoch dates work in annotation and table queries.
+func ConvertSqlTimeColumnToEpochMs(values RowValues, timeIndex int) {
+ if timeIndex >= 0 {
+ switch value := values[timeIndex].(type) {
+ case time.Time:
+ values[timeIndex] = EpochPrecisionToMs(float64(value.Unix()))
+ case *time.Time:
+ if value != nil {
+ values[timeIndex] = EpochPrecisionToMs(float64((*value).Unix()))
+ }
+ case int64:
+ values[timeIndex] = int64(EpochPrecisionToMs(float64(value)))
+ case *int64:
+ if value != nil {
+ values[timeIndex] = int64(EpochPrecisionToMs(float64(*value)))
+ }
+ case float64:
+ values[timeIndex] = EpochPrecisionToMs(value)
+ case *float64:
+ if value != nil {
+ values[timeIndex] = EpochPrecisionToMs(*value)
+ }
+ }
+ }
+}
diff --git a/pkg/tsdb/sql_engine_test.go b/pkg/tsdb/sql_engine_test.go
new file mode 100644
index 00000000000..48aac2c4d45
--- /dev/null
+++ b/pkg/tsdb/sql_engine_test.go
@@ -0,0 +1,46 @@
+package tsdb
+
+import (
+ "testing"
+ "time"
+
+ . "github.com/smartystreets/goconvey/convey"
+)
+
+func TestSqlEngine(t *testing.T) {
+ Convey("SqlEngine", t, func() {
+ Convey("Given row values with time columns when converting them", func() {
+ dt := time.Date(2018, 3, 14, 21, 20, 6, 527e6, time.UTC)
+ fixtures := make([]interface{}, 8)
+ fixtures[0] = dt
+ fixtures[1] = dt.Unix() * 1000
+ fixtures[2] = dt.Unix()
+ fixtures[3] = float64(dt.Unix() * 1000)
+ fixtures[4] = float64(dt.Unix())
+
+ var nilDt *time.Time
+ var nilInt64 *int64
+ var nilFloat64 *float64
+ fixtures[5] = nilDt
+ fixtures[6] = nilInt64
+ fixtures[7] = nilFloat64
+
+ for i := range fixtures {
+ ConvertSqlTimeColumnToEpochMs(fixtures, i)
+ }
+
+ Convey("Should convert sql time columns to epoch time in ms ", func() {
+ expected := float64(dt.Unix() * 1000)
+ So(fixtures[0].(float64), ShouldEqual, expected)
+ So(fixtures[1].(int64), ShouldEqual, expected)
+ So(fixtures[2].(int64), ShouldEqual, expected)
+ So(fixtures[3].(float64), ShouldEqual, expected)
+ So(fixtures[4].(float64), ShouldEqual, expected)
+
+ So(fixtures[5], ShouldBeNil)
+ So(fixtures[6], ShouldBeNil)
+ So(fixtures[7], ShouldBeNil)
+ })
+ })
+ })
+}
diff --git a/pkg/tsdb/time_range.go b/pkg/tsdb/time_range.go
index fd797bf731a..fd0cb3f8e82 100644
--- a/pkg/tsdb/time_range.go
+++ b/pkg/tsdb/time_range.go
@@ -88,3 +88,13 @@ func (tr *TimeRange) ParseTo() (time.Time, error) {
return time.Time{}, fmt.Errorf("cannot parse to value %s", tr.To)
}
+
+// EpochPrecisionToMs converts epoch precision to millisecond, if needed.
+// Only seconds to milliseconds supported right now
+func EpochPrecisionToMs(value float64) float64 {
+ if int64(value)/1e10 == 0 {
+ return float64(value * 1e3)
+ }
+
+ return float64(value)
+}
diff --git a/public/app/plugins/datasource/mssql/partials/annotations.editor.html b/public/app/plugins/datasource/mssql/partials/annotations.editor.html
index ecdffd92d1e..75eaa3ed1d9 100644
--- a/public/app/plugins/datasource/mssql/partials/annotations.editor.html
+++ b/public/app/plugins/datasource/mssql/partials/annotations.editor.html
@@ -20,23 +20,22 @@
Annotation Query Format
An annotation is an event that is overlayed on top of graphs. The query can have up to three columns per row, the time column is mandatory. Annotation rendering is expensive so it is important to limit the number of rows returned.
-- column with alias: time for the annotation event time (in UTC). Use unix timestamp in seconds or any native date data type.
+- column with alias: time for the annotation event time. Use epoch time or any native date data type.
- column with alias: text for the annotation text.
- column with alias: tags for annotation tags. This is a comma separated string of tags e.g. 'tag1,tag2'.
Macros:
- $__time(column) -> column AS time
-- $__utcTime(column) -> DATEADD(second, DATEDIFF(second, GETDATE(), GETUTCDATE()), column) AS time
-- $__timeEpoch(column) -> DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second, GETDATE(), GETUTCDATE()), column) ) AS time
-- $__timeFilter(column) -> column > DATEADD(s, 1492750877+DATEDIFF(second, GETUTCDATE(), GETDATE()), '1970-01-01') AND column < DATEADD(s, 1492750877+DATEDIFF(second, GETUTCDATE(), GETDATE()), '1970-01-01')
-- $__unixEpochFilter(column) -> column > 1492750877 AND column < 1492750877
+- $__timeEpoch(column) -> DATEDIFF(second, '1970-01-01', column) AS time
+- $__timeFilter(column) -> column >= DATEADD(s, 18446744066914186738, '1970-01-01') AND column &t;= DATEADD(s, 18446744066914187038, '1970-01-01')
+- $__unixEpochFilter(column) -> column >= 1492750877 AND column <= 1492750877
Or build your own conditionals using these macros which just return the values:
-- $__timeFrom() -> DATEADD(second, 1492750877+DATEDIFF(second, GETUTCDATE(), GETDATE()), '1970-01-01')
-- $__timeTo() -> DATEADD(second, 1492750877+DATEDIFF(second, GETUTCDATE(), GETDATE()), '1970-01-01')
-- $__unixEpochFrom() -> 1492750877
-- $__unixEpochTo() -> 1492750877
+- $__timeFrom() -> DATEADD(second, 1492750877, '1970-01-01')
+- $__timeTo() -> DATEADD(second, 1492750877, '1970-01-01')
+- $__unixEpochFrom() -> 1492750877
+- $__unixEpochTo() -> 1492750877
diff --git a/public/app/plugins/datasource/mssql/partials/query.editor.html b/public/app/plugins/datasource/mssql/partials/query.editor.html
index e8f44c8c9f8..c7dc030be6e 100644
--- a/public/app/plugins/datasource/mssql/partials/query.editor.html
+++ b/public/app/plugins/datasource/mssql/partials/query.editor.html
@@ -48,15 +48,22 @@ Table:
Macros:
- $__time(column) -> column AS time
-- $__utcTime(column) -> DATEADD(second, DATEDIFF(second, GETDATE(), GETUTCDATE()), column) AS time
-- $__timeEpoch(column) -> DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second, GETDATE(), GETUTCDATE()), column) ) AS time
-- $__timeFilter(column) -> column > DATEADD(s, 1492750877+DATEDIFF(second, GETUTCDATE(), GETDATE()), '1970-01-01') AND column < DATEADD(s, 1492750877+DATEDIFF(second, GETUTCDATE(), GETDATE()), '1970-01-01')
-- $__unixEpochFilter(column) -> column > 1492750877 AND column < 1492750877
-- $__timeGroup(column, '5m'[, fillvalue]) -> cast(cast(DATEDIFF(second, {d '1970-01-01'}, DATEADD(second, DATEDIFF(second, GETDATE(), GETUTCDATE()), column))/300 as int)*300 as int). Providing a fillValue of NULL or floating value will automatically fill empty series in timerange with that value.
+- $__timeEpoch(column) -> DATEDIFF(second, '1970-01-01', column) AS time
+- $__timeFilter(column) -> column >= DATEADD(s, 18446744066914186738, '1970-01-01') AND column &t;= DATEADD(s, 18446744066914187038, '1970-01-01')
+- $__unixEpochFilter(column) -> column >= 1492750877 AND column <= 1492750877
+- $__timeGroup(column, '5m'[, fillvalue]) -> CAST(ROUND(DATEDIFF(second, '1970-01-01', column)/300.0, 0) as bigint)*300. Providing a fillValue of NULL or floating value will automatically fill empty series in timerange with that value.
+
+Example of group by and order by with $__timeGroup:
+SELECT
+ $__timeGroup(date_time_col, '1h') AS time,
+ sum(value) as value
+FROM yourtable
+GROUP BY $__timeGroup(date_time_col, '1h')
+ORDER BY 1
Or build your own conditionals using these macros which just return the values:
-- $__timeFrom() -> DATEADD(second, 1492750877+DATEDIFF(second, GETUTCDATE(), GETDATE()), '1970-01-01')
-- $__timeTo() -> DATEADD(second, 1492750877+DATEDIFF(second, GETUTCDATE(), GETDATE()), '1970-01-01')
+- $__timeFrom() -> DATEADD(second, 1492750877, '1970-01-01')
+- $__timeTo() -> DATEADD(second, 1492750877, '1970-01-01')
- $__unixEpochFrom() -> 1492750877
- $__unixEpochTo() -> 1492750877
diff --git a/public/app/plugins/datasource/mssql/response_parser.ts b/public/app/plugins/datasource/mssql/response_parser.ts
index b7d96d820cb..b6f538707b0 100644
--- a/public/app/plugins/datasource/mssql/response_parser.ts
+++ b/public/app/plugins/datasource/mssql/response_parser.ts
@@ -128,7 +128,7 @@ export default class ResponseParser {
const row = table.rows[i];
list.push({
annotation: options.annotation,
- time: Math.floor(row[timeColumnIndex]) * 1000,
+ time: Math.floor(row[timeColumnIndex]),
text: row[textColumnIndex],
tags: row[tagsColumnIndex] ? row[tagsColumnIndex].trim().split(/\s*,\s*/) : [],
});
diff --git a/public/app/plugins/datasource/mssql/specs/datasource_specs.ts b/public/app/plugins/datasource/mssql/specs/datasource.jest.ts
similarity index 67%
rename from public/app/plugins/datasource/mssql/specs/datasource_specs.ts
rename to public/app/plugins/datasource/mssql/specs/datasource.jest.ts
index 923b7fd2fc1..dd2d4a60cec 100644
--- a/public/app/plugins/datasource/mssql/specs/datasource_specs.ts
+++ b/public/app/plugins/datasource/mssql/specs/datasource.jest.ts
@@ -1,26 +1,21 @@
-import { describe, beforeEach, it, expect, angularMocks } from 'test/lib/common';
import moment from 'moment';
-import helpers from 'test/specs/helpers';
import { MssqlDatasource } from '../datasource';
+import { TemplateSrvStub } from 'test/specs/helpers';
import { CustomVariable } from 'app/features/templating/custom_variable';
+import q from 'q';
describe('MSSQLDatasource', function() {
- var ctx = new helpers.ServiceTestContext();
- var instanceSettings = { name: 'mssql' };
+ const ctx: any = {
+ backendSrv: {},
+ templateSrv: new TemplateSrvStub(),
+ };
- beforeEach(angularMocks.module('grafana.core'));
- beforeEach(angularMocks.module('grafana.services'));
- beforeEach(ctx.providePhase(['backendSrv']));
+ beforeEach(function() {
+ ctx.$q = q;
+ ctx.instanceSettings = { name: 'mssql' };
- beforeEach(
- angularMocks.inject(function($q, $rootScope, $httpBackend, $injector) {
- ctx.$q = $q;
- ctx.$httpBackend = $httpBackend;
- ctx.$rootScope = $rootScope;
- ctx.ds = $injector.instantiate(MssqlDatasource, { instanceSettings: instanceSettings });
- $httpBackend.when('GET', /\.html$/).respond('');
- })
- );
+ ctx.ds = new MssqlDatasource(ctx.instanceSettings, ctx.backendSrv, ctx.$q, ctx.templateSrv);
+ });
describe('When performing annotationQuery', function() {
let results;
@@ -46,9 +41,9 @@ describe('MSSQLDatasource', function() {
{
columns: [{ text: 'time' }, { text: 'text' }, { text: 'tags' }],
rows: [
- [1432288355, 'some text', 'TagA,TagB'],
- [1432288390, 'some text2', ' TagB , TagC'],
- [1432288400, 'some text3'],
+ [1521545610656, 'some text', 'TagA,TagB'],
+ [1521546251185, 'some text2', ' TagB , TagC'],
+ [1521546501378, 'some text3'],
],
},
],
@@ -56,27 +51,27 @@ describe('MSSQLDatasource', function() {
},
};
- beforeEach(function() {
- ctx.backendSrv.datasourceRequest = function(options) {
+ beforeEach(() => {
+ ctx.backendSrv.datasourceRequest = options => {
return ctx.$q.when({ data: response, status: 200 });
};
- ctx.ds.annotationQuery(options).then(function(data) {
+
+ return ctx.ds.annotationQuery(options).then(data => {
results = data;
});
- ctx.$rootScope.$apply();
});
it('should return annotation list', function() {
- expect(results.length).to.be(3);
+ expect(results.length).toBe(3);
- expect(results[0].text).to.be('some text');
- expect(results[0].tags[0]).to.be('TagA');
- expect(results[0].tags[1]).to.be('TagB');
+ expect(results[0].text).toBe('some text');
+ expect(results[0].tags[0]).toBe('TagA');
+ expect(results[0].tags[1]).toBe('TagB');
- expect(results[1].tags[0]).to.be('TagB');
- expect(results[1].tags[1]).to.be('TagC');
+ expect(results[1].tags[0]).toBe('TagB');
+ expect(results[1].tags[1]).toBe('TagC');
- expect(results[2].tags.length).to.be(0);
+ expect(results[2].tags.length).toBe(0);
});
});
@@ -104,16 +99,16 @@ describe('MSSQLDatasource', function() {
ctx.backendSrv.datasourceRequest = function(options) {
return ctx.$q.when({ data: response, status: 200 });
};
- ctx.ds.metricFindQuery(query).then(function(data) {
+
+ return ctx.ds.metricFindQuery(query).then(function(data) {
results = data;
});
- ctx.$rootScope.$apply();
});
it('should return list of all column values', function() {
- expect(results.length).to.be(6);
- expect(results[0].text).to.be('aTitle');
- expect(results[5].text).to.be('some text3');
+ expect(results.length).toBe(6);
+ expect(results[0].text).toBe('aTitle');
+ expect(results[5].text).toBe('some text3');
});
});
@@ -141,18 +136,18 @@ describe('MSSQLDatasource', function() {
ctx.backendSrv.datasourceRequest = function(options) {
return ctx.$q.when({ data: response, status: 200 });
};
- ctx.ds.metricFindQuery(query).then(function(data) {
+
+ return ctx.ds.metricFindQuery(query).then(function(data) {
results = data;
});
- ctx.$rootScope.$apply();
});
it('should return list of as text, value', function() {
- expect(results.length).to.be(3);
- expect(results[0].text).to.be('aTitle');
- expect(results[0].value).to.be('value1');
- expect(results[2].text).to.be('aTitle3');
- expect(results[2].value).to.be('value3');
+ expect(results.length).toBe(3);
+ expect(results[0].text).toBe('aTitle');
+ expect(results[0].value).toBe('value1');
+ expect(results[2].text).toBe('aTitle3');
+ expect(results[2].value).toBe('value3');
});
});
@@ -180,16 +175,16 @@ describe('MSSQLDatasource', function() {
ctx.backendSrv.datasourceRequest = function(options) {
return ctx.$q.when({ data: response, status: 200 });
};
- ctx.ds.metricFindQuery(query).then(function(data) {
+
+ return ctx.ds.metricFindQuery(query).then(function(data) {
results = data;
});
- ctx.$rootScope.$apply();
});
it('should return list of unique keys', function() {
- expect(results.length).to.be(1);
- expect(results[0].text).to.be('aTitle');
- expect(results[0].value).to.be('same');
+ expect(results.length).toBe(1);
+ expect(results[0].text).toBe('aTitle');
+ expect(results[0].value).toBe('same');
});
});
@@ -200,33 +195,33 @@ describe('MSSQLDatasource', function() {
describe('and value is a string', () => {
it('should return an unquoted value', () => {
- expect(ctx.ds.interpolateVariable('abc', ctx.variable)).to.eql('abc');
+ expect(ctx.ds.interpolateVariable('abc', ctx.variable)).toEqual('abc');
});
});
describe('and value is a number', () => {
it('should return an unquoted value', () => {
- expect(ctx.ds.interpolateVariable(1000, ctx.variable)).to.eql(1000);
+ expect(ctx.ds.interpolateVariable(1000, ctx.variable)).toEqual(1000);
});
});
describe('and value is an array of strings', () => {
it('should return comma separated quoted values', () => {
- expect(ctx.ds.interpolateVariable(['a', 'b', 'c'], ctx.variable)).to.eql("'a','b','c'");
+ expect(ctx.ds.interpolateVariable(['a', 'b', 'c'], ctx.variable)).toEqual("'a','b','c'");
});
});
describe('and variable allows multi-value and value is a string', () => {
it('should return a quoted value', () => {
ctx.variable.multi = true;
- expect(ctx.ds.interpolateVariable('abc', ctx.variable)).to.eql("'abc'");
+ expect(ctx.ds.interpolateVariable('abc', ctx.variable)).toEqual("'abc'");
});
});
describe('and variable allows all and value is a string', () => {
it('should return a quoted value', () => {
ctx.variable.includeAll = true;
- expect(ctx.ds.interpolateVariable('abc', ctx.variable)).to.eql("'abc'");
+ expect(ctx.ds.interpolateVariable('abc', ctx.variable)).toEqual("'abc'");
});
});
});
diff --git a/public/app/plugins/datasource/mysql/partials/annotations.editor.html b/public/app/plugins/datasource/mysql/partials/annotations.editor.html
index b34eff5b011..d142e091fed 100644
--- a/public/app/plugins/datasource/mysql/partials/annotations.editor.html
+++ b/public/app/plugins/datasource/mysql/partials/annotations.editor.html
@@ -18,15 +18,16 @@
Annotation Query Format
-An annotation is an event that is overlayed on top of graphs. The query can have up to four columns per row, the time_sec column is mandatory. Annotation rendering is expensive so it is important to limit the number of rows returned.
+An annotation is an event that is overlayed on top of graphs. The query can have up to three columns per row, the time or time_sec column is mandatory. Annotation rendering is expensive so it is important to limit the number of rows returned.
-- column with alias: time_sec for the annotation event. Format is UTC in seconds, use UNIX_TIMESTAMP(column)
+- column with alias: time or time_sec for the annotation event time. Use epoch time or any native date data type.
- column with alias: text for the annotation text
- column with alias: tags for annotation tags. This is a comma separated string of tags e.g. 'tag1,tag2'
Macros:
-- $__time(column) -> UNIX_TIMESTAMP(column) as time_sec
+- $__time(column) -> UNIX_TIMESTAMP(column) as time (or as time_sec)
+- $__timeEpoch(column) -> UNIX_TIMESTAMP(column) as time (or as time_sec)
- $__timeFilter(column) -> UNIX_TIMESTAMP(time_date_time) > 1492750877 AND UNIX_TIMESTAMP(time_date_time) < 1492750877
- $__unixEpochFilter(column) -> time_unix_epoch > 1492750877 AND time_unix_epoch < 1492750877
diff --git a/public/app/plugins/datasource/mysql/partials/query.editor.html b/public/app/plugins/datasource/mysql/partials/query.editor.html
index 22d64c9190f..9acf32405c1 100644
--- a/public/app/plugins/datasource/mysql/partials/query.editor.html
+++ b/public/app/plugins/datasource/mysql/partials/query.editor.html
@@ -38,15 +38,16 @@
Time series:
-- return column named time_sec (UTC in seconds), use UNIX_TIMESTAMP(column)
-- return column named value for the time point value
-- return column named metric to represent the series name
+- return column named time or time_sec (in UTC), as a unix time stamp or any sql native date data type. You can use the macros below.
+- return column(s) with numeric datatype as values
+- (Optional: return column named metric to represent the series name. If no column named metric is found the column name of the value column is used as series name)
Table:
- return any set of columns
Macros:
- $__time(column) -> UNIX_TIMESTAMP(column) as time_sec
+- $__timeEpoch(column) -> UNIX_TIMESTAMP(column) as time_sec
- $__timeFilter(column) -> UNIX_TIMESTAMP(time_date_time) ≥ 1492750877 AND UNIX_TIMESTAMP(time_date_time) ≤ 1492750877
- $__unixEpochFilter(column) -> time_unix_epoch > 1492750877 AND time_unix_epoch < 1492750877
- $__timeGroup(column,'5m') -> cast(cast(UNIX_TIMESTAMP(column)/(300) as signed)*300 as signed)
diff --git a/public/app/plugins/datasource/mysql/response_parser.ts b/public/app/plugins/datasource/mysql/response_parser.ts
index 50683578035..e5d8ab79f2a 100644
--- a/public/app/plugins/datasource/mysql/response_parser.ts
+++ b/public/app/plugins/datasource/mysql/response_parser.ts
@@ -113,7 +113,7 @@ export default class ResponseParser {
let tagsColumnIndex = -1;
for (let i = 0; i < table.columns.length; i++) {
- if (table.columns[i].text === 'time_sec') {
+ if (table.columns[i].text === 'time_sec' || table.columns[i].text === 'time') {
timeColumnIndex = i;
} else if (table.columns[i].text === 'title') {
return this.$q.reject({
@@ -137,7 +137,7 @@ export default class ResponseParser {
const row = table.rows[i];
list.push({
annotation: options.annotation,
- time: Math.floor(row[timeColumnIndex]) * 1000,
+ time: Math.floor(row[timeColumnIndex]),
text: row[textColumnIndex] ? row[textColumnIndex].toString() : '',
tags: row[tagsColumnIndex] ? row[tagsColumnIndex].trim().split(/\s*,\s*/) : [],
});
diff --git a/public/app/plugins/datasource/postgres/partials/annotations.editor.html b/public/app/plugins/datasource/postgres/partials/annotations.editor.html
index b56f7523087..09232d6f8ed 100644
--- a/public/app/plugins/datasource/postgres/partials/annotations.editor.html
+++ b/public/app/plugins/datasource/postgres/partials/annotations.editor.html
@@ -18,15 +18,16 @@
Annotation Query Format
-An annotation is an event that is overlayed on top of graphs. The query can have up to four columns per row, the time column is mandatory. Annotation rendering is expensive so it is important to limit the number of rows returned.
+An annotation is an event that is overlayed on top of graphs. The query can have up to three columns per row, the time column is mandatory. Annotation rendering is expensive so it is important to limit the number of rows returned.
-- column with alias: time for the annotation event. Format is UTC in seconds, use extract(epoch from column) as "time"
+- column with alias: time for the annotation event time. Use epoch time or any native date data type.
- column with alias: text for the annotation text
- column with alias: tags for annotation tags. This is a comma separated string of tags e.g. 'tag1,tag2'
Macros:
- $__time(column) -> column as "time"
+- $__timeEpoch -> extract(epoch from column) as "time"
- $__timeFilter(column) -> column ≥ to_timestamp(1492750877) AND column ≤ to_timestamp(1492750877)
- $__unixEpochFilter(column) -> column > 1492750877 AND column < 1492750877
diff --git a/public/app/plugins/datasource/postgres/response_parser.ts b/public/app/plugins/datasource/postgres/response_parser.ts
index 620aba5fa7e..ebc9598468b 100644
--- a/public/app/plugins/datasource/postgres/response_parser.ts
+++ b/public/app/plugins/datasource/postgres/response_parser.ts
@@ -134,7 +134,7 @@ export default class ResponseParser {
const row = table.rows[i];
list.push({
annotation: options.annotation,
- time: Math.floor(row[timeColumnIndex]) * 1000,
+ time: Math.floor(row[timeColumnIndex]),
title: row[titleColumnIndex],
text: row[textColumnIndex],
tags: row[tagsColumnIndex] ? row[tagsColumnIndex].trim().split(/\s*,\s*/) : [],
diff --git a/public/app/plugins/panel/graph/legend.ts b/public/app/plugins/panel/graph/legend.ts
index 0c8852bf55a..d1186ae0b1e 100644
--- a/public/app/plugins/panel/graph/legend.ts
+++ b/public/app/plugins/panel/graph/legend.ts
@@ -15,6 +15,7 @@ module.directive('graphLegend', function(popoverSrv, $timeout) {
var seriesList;
var i;
var legendScrollbar;
+ const legendRightDefaultWidth = 10;
scope.$on('$destroy', function() {
if (legendScrollbar) {
@@ -111,6 +112,7 @@ module.directive('graphLegend', function(popoverSrv, $timeout) {
}
function render() {
+ let legendWidth = elem.width();
if (!ctrl.panel.legend.show) {
elem.empty();
firstRender = true;
@@ -163,7 +165,7 @@ module.directive('graphLegend', function(popoverSrv, $timeout) {
}
// render first time for getting proper legend height
- if (!panel.legend.rightSide) {
+ if (!panel.legend.rightSide || (panel.legend.rightSide && legendWidth !== legendRightDefaultWidth)) {
renderLegendElement(tableHeaderElem);
elem.empty();
}
@@ -227,6 +229,8 @@ module.directive('graphLegend', function(popoverSrv, $timeout) {
}
function renderLegendElement(tableHeaderElem) {
+ let legendWidth = elem.width();
+
var seriesElements = renderSeriesLegendElements();
if (panel.legend.alignAsTable) {
@@ -238,7 +242,7 @@ module.directive('graphLegend', function(popoverSrv, $timeout) {
elem.append(seriesElements);
}
- if (!panel.legend.rightSide) {
+ if (!panel.legend.rightSide || (panel.legend.rightSide && legendWidth !== legendRightDefaultWidth)) {
addScrollbar();
} else {
destroyScrollbar();
@@ -263,6 +267,7 @@ module.directive('graphLegend', function(popoverSrv, $timeout) {
function destroyScrollbar() {
if (legendScrollbar) {
legendScrollbar.destroy();
+ legendScrollbar = undefined;
}
}
},
diff --git a/public/sass/components/_panel_graph.scss b/public/sass/components/_panel_graph.scss
index 716778096d6..e15cd576367 100644
--- a/public/sass/components/_panel_graph.scss
+++ b/public/sass/components/_panel_graph.scss
@@ -4,24 +4,26 @@
height: 100%;
&--legend-right {
- flex-direction: row;
+ @include media-breakpoint-up(sm) {
+ flex-direction: row;
- .graph-legend {
- flex: 0 1 10px;
- max-height: 100%;
- }
+ .graph-legend {
+ flex: 0 1 10px;
+ max-height: 100%;
+ }
- .graph-legend-series {
- display: block;
- padding-left: 0px;
- }
+ .graph-legend-series {
+ display: block;
+ padding-left: 0px;
+ }
- .graph-legend-table {
- width: auto;
- }
+ .graph-legend-table {
+ width: auto;
+ }
- .graph-legend-table .graph-legend-series {
- display: table-row;
+ .graph-legend-table .graph-legend-series {
+ display: table-row;
+ }
}
}
}