From 74b259d14a2396c0bff453626de278935f1b9b23 Mon Sep 17 00:00:00 2001 From: Hardikl <83282894+Hardikl@users.noreply.github.com> Date: Tue, 10 Sep 2024 17:23:42 +0530 Subject: [PATCH] feat: adding panels for OntapS3SVM object (#3134) * feat: adding new dashboard for OntapS3SVM object --- cmd/collectors/ems/ems.go | 2 +- cmd/collectors/keyperf/keyperf.go | 2 +- cmd/collectors/rest/plugins/health/health.go | 1 + cmd/collectors/restperf/restperf.go | 2 +- cmd/collectors/zapi/plugins/qtree/qtree.go | 2 +- cmd/collectors/zapiperf/zapiperf.go | 2 +- cmd/tools/grafana/grafana_test.go | 2 +- conf/restperf/9.14.1/ontap_s3_svm.yaml | 14 +- conf/zapiperf/cdot/9.8.0/ontap_s3_svm.yaml | 14 +- grafana/dashboards/cmode/s3ObjectStorage.json | 756 +++++++++++++++++- integration/test/dashboard_json_test.go | 1 + pkg/matrix/matrix.go | 2 +- 12 files changed, 777 insertions(+), 23 deletions(-) diff --git a/cmd/collectors/ems/ems.go b/cmd/collectors/ems/ems.go index 5af33acdb..36fb328c9 100644 --- a/cmd/collectors/ems/ems.go +++ b/cmd/collectors/ems/ems.go @@ -344,7 +344,7 @@ func (e *Ems) PollInstance() (map[string]*matrix.Matrix, error) { // update metadata for collector logs _ = e.Metadata.LazySetValueInt64("api_time", "instance", apiD.Microseconds()) _ = e.Metadata.LazySetValueInt64("parse_time", "instance", time.Since(parseT).Microseconds()) - _ = e.Metadata.LazySetValueUint64("instances", "instance", uint64(bookendCacheSize)) + _ = e.Metadata.LazySetValueUint64("instances", "instance", uint64(bookendCacheSize)) //nolint:gosec return nil, nil } diff --git a/cmd/collectors/keyperf/keyperf.go b/cmd/collectors/keyperf/keyperf.go index f9a5d48d6..2be240a35 100644 --- a/cmd/collectors/keyperf/keyperf.go +++ b/cmd/collectors/keyperf/keyperf.go @@ -441,7 +441,7 @@ func (kp *KeyPerf) pollData( calcD := time.Since(calcStart) _ = kp.Metadata.LazySetValueUint64("instances", "data", uint64(len(curMat.GetInstances()))) _ = kp.Metadata.LazySetValueInt64("calc_time", "data", calcD.Microseconds()) - _ = kp.Metadata.LazySetValueUint64("skips", "data", uint64(totalSkips)) + _ = kp.Metadata.LazySetValueUint64("skips", "data", uint64(totalSkips)) //nolint:gosec // store cache for next poll kp.Matrix[kp.Object] = cachedData diff --git a/cmd/collectors/rest/plugins/health/health.go b/cmd/collectors/rest/plugins/health/health.go index 776ad237a..80d91d6b8 100644 --- a/cmd/collectors/rest/plugins/health/health.go +++ b/cmd/collectors/rest/plugins/health/health.go @@ -179,6 +179,7 @@ func (h *Health) Run(dataMap map[string]*matrix.Matrix) ([]*matrix.Matrix, *util Int("numResolutionInstanceCount", resolutionInstancesCount). Msg("Collected") + //nolint:gosec h.client.Metadata.PluginInstances = uint64(diskAlertCount + shelfAlertCount + supportAlertCount + nodeAlertCount + HAAlertCount + networkEthernetPortAlertCount + networkFcpPortAlertCount + networkInterfaceAlertCount + volumeRansomwareAlertCount + volumeMoveAlertCount + licenseAlertCount + resolutionInstancesCount) diff --git a/cmd/collectors/restperf/restperf.go b/cmd/collectors/restperf/restperf.go index d64fcc942..e820ec7f6 100644 --- a/cmd/collectors/restperf/restperf.go +++ b/cmd/collectors/restperf/restperf.go @@ -1272,7 +1272,7 @@ func (r *RestPerf) pollData(startTime time.Time, perfRecords []rest.PerfRecord) calcD := time.Since(calcStart) _ = r.Metadata.LazySetValueUint64("instances", "data", uint64(len(curMat.GetInstances()))) _ = r.Metadata.LazySetValueInt64("calc_time", "data", calcD.Microseconds()) - _ = r.Metadata.LazySetValueUint64("skips", "data", uint64(totalSkips)) + _ = r.Metadata.LazySetValueUint64("skips", "data", uint64(totalSkips)) //nolint:gosec // store cache for next poll r.Matrix[r.Object] = cachedData diff --git a/cmd/collectors/zapi/plugins/qtree/qtree.go b/cmd/collectors/zapi/plugins/qtree/qtree.go index d7d5495fd..59d4b202d 100644 --- a/cmd/collectors/zapi/plugins/qtree/qtree.go +++ b/cmd/collectors/zapi/plugins/qtree/qtree.go @@ -215,7 +215,7 @@ func (q *Qtree) Run(dataMap map[string]*matrix.Matrix) ([]*matrix.Matrix, *util. } } - q.client.Metadata.PluginInstances = uint64(quotaIndex) + q.client.Metadata.PluginInstances = uint64(quotaIndex) //nolint:gosec q.Logger.Info(). Int("numQuotas", quotaIndex). diff --git a/cmd/collectors/zapiperf/zapiperf.go b/cmd/collectors/zapiperf/zapiperf.go index 459ce0896..85d16b240 100644 --- a/cmd/collectors/zapiperf/zapiperf.go +++ b/cmd/collectors/zapiperf/zapiperf.go @@ -887,7 +887,7 @@ func (z *ZapiPerf) PollData() (map[string]*matrix.Matrix, error) { calcD := time.Since(calcStart) _ = z.Metadata.LazySetValueInt64("calc_time", "data", calcD.Microseconds()) - _ = z.Metadata.LazySetValueUint64("skips", "data", uint64(totalSkips)) + _ = z.Metadata.LazySetValueUint64("skips", "data", uint64(totalSkips)) //nolint:gosec // store cache for next poll z.Matrix[z.Object] = cachedData diff --git a/cmd/tools/grafana/grafana_test.go b/cmd/tools/grafana/grafana_test.go index 548fdcea2..eb6617afc 100644 --- a/cmd/tools/grafana/grafana_test.go +++ b/cmd/tools/grafana/grafana_test.go @@ -99,7 +99,7 @@ func TestAddPrefixToMetricNames(t *testing.T) { } newExpressions = readExprs(updatedData) - for i := range len(newExpressions) { + for i := range newExpressions { if newExpressions[i] != prefix+oldExpressions[i] { t.Errorf("path: %s \nExpected: [%s]\n Got: [%s]", path, prefix+oldExpressions[i], newExpressions[i]) } diff --git a/conf/restperf/9.14.1/ontap_s3_svm.yaml b/conf/restperf/9.14.1/ontap_s3_svm.yaml index 5e194c6fd..c59b5f504 100644 --- a/conf/restperf/9.14.1/ontap_s3_svm.yaml +++ b/conf/restperf/9.14.1/ontap_s3_svm.yaml @@ -135,7 +135,13 @@ counters: - upload_part_rate - upload_part_total -export_options: - instance_keys: - - node - - svm +plugins: + - Aggregator: + # plugin will create summary/average for each ontaps3_svm object + # any names after the object names will be treated as + # label names that will be added to instances + - svm<>ontaps3_svm + +# only export svm aggregations from plugin +# set this true or comment, to get data for each ontaps3_svm +export_data: false diff --git a/conf/zapiperf/cdot/9.8.0/ontap_s3_svm.yaml b/conf/zapiperf/cdot/9.8.0/ontap_s3_svm.yaml index d703d492e..1ea5e2292 100644 --- a/conf/zapiperf/cdot/9.8.0/ontap_s3_svm.yaml +++ b/conf/zapiperf/cdot/9.8.0/ontap_s3_svm.yaml @@ -134,7 +134,13 @@ counters: - upload_part_total - vserver_name => svm -export_options: - instance_keys: - - node - - svm +plugins: + - Aggregator: + # plugin will create summary/average for each ontaps3_svm object + # any names after the object names will be treated as + # label names that will be added to instances + - svm<>ontaps3_svm + +# only export svm aggregations from plugin +# set this true or comment, to get data for each ontaps3_svm +export_data: false diff --git a/grafana/dashboards/cmode/s3ObjectStorage.json b/grafana/dashboards/cmode/s3ObjectStorage.json index 0795cdaa0..3e1f61e35 100644 --- a/grafana/dashboards/cmode/s3ObjectStorage.json +++ b/grafana/dashboards/cmode/s3ObjectStorage.json @@ -101,14 +101,14 @@ { "datasource": "${DS_PROMETHEUS}", "gridPos": { - "h": 2, + "h": 4, "w": 24, "x": 0, "y": 1 }, "id": 107, "options": { - "content": "This dashboard requires ONTAP 9.7+. One action is required to use this dashboard:
1. On Harvest, enable the Rest collector for OntapS3 template in your `harvest.yml` config for configuration metrics", + "content": "This dashboard requires ONTAP 9.7+. Three actions are required to use this dashboard:
1. If you are using the Rest collector, then you must enable the OntapS3 counters in `$HARVEST/conf/rest/default.yaml`.
2. If you are using the ZapiPerf collector, then you must enable the OntapS3SVM counters in `$HARVEST/conf/zapiperf/default.yaml`.
3. If you are using the RestPerf collector, then you must enable the OntapS3SVM counters in `$HARVEST/conf/restperf/default.yaml`.
Please see https://github.com/NetApp/harvest/discussions/3137 for more details.", "mode": "markdown" }, "pluginVersion": "8.1.8", @@ -120,7 +120,7 @@ "h": 1, "w": 24, "x": 0, - "y": 3 + "y": 5 }, "id": 16, "title": "Bucket", @@ -352,7 +352,7 @@ "h": 8, "w": 24, "x": 0, - "y": 4 + "y": 6 }, "id": 5, "options": { @@ -564,7 +564,7 @@ "h": 11, "w": 12, "x": 0, - "y": 12 + "y": 14 }, "id": 127, "options": { @@ -682,7 +682,7 @@ "h": 11, "w": 12, "x": 12, - "y": 12 + "y": 14 }, "id": 128, "options": { @@ -838,7 +838,7 @@ "h": 8, "w": 24, "x": 0, - "y": 23 + "y": 25 }, "id": 6, "options": { @@ -910,6 +910,746 @@ } ], "type": "table" + }, + { + "collapsed": true, + "datasource": "${DS_PROMETHEUS}", + "gridPos": { + "h": 1, + "w": 24, + "x": 0, + "y": 33 + }, + "id": 17, + "panels": [ + { + "datasource": "${DS_PROMETHEUS}", + "description": "This panel displays Requests and Connections counts at each SVM.", + "fieldConfig": { + "defaults": { + "color": { + "mode": "thresholds" + }, + "custom": { + "align": "auto", + "cellOptions": { + "type": "auto" + }, + "filterable": true, + "inspect": false + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + }, + { + "color": "red", + "value": 80 + } + ] + }, + "unit": "short", + "unitScale": true + }, + "overrides": [ + { + "matcher": { + "id": "byName", + "options": "datacenter" + }, + "properties": [ + { + "id": "displayName", + "value": "Datacenter" + }, + { + "id": "links", + "value": [ + { + "targetBlank": true, + "title": "", + "url": "/d/cdot-datacenter/ontap-datacenter?orgId=1&${__url_time_range}&var-Datacenter=${__value.raw}" + } + ] + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "cluster" + }, + "properties": [ + { + "id": "displayName", + "value": "Cluster" + }, + { + "id": "links", + "value": [ + { + "targetBlank": true, + "title": "", + "url": "/d/cdot-cluster/ontap-cluster?orgId=1&${Datacenter:queryparam}&${__url_time_range}&var-Cluster=${__value.raw}" + } + ] + } + ] + }, + { + "matcher": { + "id": "byName", + "options": "svm" + }, + "properties": [ + { + "id": "displayName", + "value": "SVM" + }, + { + "id": "links", + "value": [ + { + "targetBlank": true, + "title": "", + "url": "/d/cdot-svm/ontap-svm?orgId=1&${Datacenter:queryparam}&${Cluster:queryparam}&${__url_time_range}&var-SVM=${__value.raw}" + } + ] + } + ] + } + ] + }, + "gridPos": { + "h": 10, + "w": 24, + "x": 0, + "y": 5 + }, + "id": 137, + "options": { + "cellHeight": "sm", + "footer": { + "countRows": false, + "fields": "", + "reducer": [ + "sum" + ], + "show": false + }, + "showHeader": true, + "sortBy": [ + { + "desc": false, + "displayName": "SVM" + } + ] + }, + "pluginVersion": "10.3.1", + "targets": [ + { + "editorMode": "code", + "exemplar": false, + "expr": "sum by (datacenter, cluster, svm)(ontaps3_svm_requests{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"})", + "format": "table", + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "A" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "sum by (datacenter, cluster, svm)(ontaps3_svm_connections{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"})", + "format": "table", + "hide": false, + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "B" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "sum by (datacenter, cluster, svm)(ontaps3_svm_max_requests_outstanding{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"})", + "format": "table", + "hide": false, + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "C" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "sum by (datacenter, cluster, svm)(ontaps3_svm_max_connected_connections{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"})", + "format": "table", + "hide": false, + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "D" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "sum by (datacenter, cluster, svm)(ontaps3_svm_max_cmds_per_connection{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"})", + "format": "table", + "hide": false, + "instant": true, + "interval": "", + "legendFormat": "", + "refId": "E" + } + ], + "title": "Requests & Connections stats", + "transformations": [ + { + "id": "filterFieldsByName", + "options": { + "include": { + "names": [ + "Value #A", + "Value #B", + "Value #C", + "Value #D", + "Value #E", + "svm", + "cluster", + "datacenter" + ] + } + } + }, + { + "id": "merge", + "options": {} + }, + { + "id": "organize", + "options": { + "excludeByName": {}, + "includeByName": {}, + "indexByName": { + "Value #A": 3, + "Value #B": 4, + "Value #C": 5, + "Value #D": 6, + "Value #E": 7, + "cluster": 1, + "datacenter": 0, + "svm": 2 + }, + "renameByName": { + "Value": "Used space", + "Value #A": "Requests", + "Value #B": "Connections", + "Value #C": "Max Outstanding Requests", + "Value #D": "Max Connections", + "Value #E": "Max Commands Per Connections" + } + } + } + ], + "type": "table" + }, + { + "datasource": "${DS_PROMETHEUS}", + "description": "Number of GET/PUT/HEAD/DELETE SVM operations.", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "Operations", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 10, + "gradientMode": "opacity", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": true, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "short", + "unitScale": true + }, + "overrides": [] + }, + "gridPos": { + "h": 11, + "w": 12, + "x": 0, + "y": 15 + }, + "id": 132, + "options": { + "legend": { + "calcs": [ + "mean", + "lastNotNull", + "max" + ], + "displayMode": "table", + "placement": "bottom", + "showLegend": true, + "sortBy": "Last *", + "sortDesc": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "pluginVersion": "8.1.8", + "targets": [ + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_get_object_total{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_get_object_total{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - GET", + "range": true, + "refId": "A" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_put_object_total{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_put_object_total{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - PUT", + "range": true, + "refId": "B" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_head_object_total{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_head_object_total{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - HEAD", + "range": true, + "refId": "C" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_delete_object_total{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_delete_object_total{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - DELETE", + "range": true, + "refId": "D" + } + ], + "title": "Top $TopResources SVMs by Operations", + "transformations": [], + "type": "timeseries" + }, + { + "datasource": "${DS_PROMETHEUS}", + "description": "Number of GET/PUT/HEAD/DELETE SVM operations per second.", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "Operations/sec", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 10, + "gradientMode": "opacity", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": true, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "decimals": 0, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "iops", + "unitScale": true + }, + "overrides": [] + }, + "gridPos": { + "h": 11, + "w": 12, + "x": 12, + "y": 15 + }, + "id": 129, + "options": { + "legend": { + "calcs": [ + "mean", + "lastNotNull", + "max" + ], + "displayMode": "table", + "placement": "bottom", + "showLegend": true, + "sortBy": "Last *", + "sortDesc": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "pluginVersion": "8.1.8", + "targets": [ + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_get_object_rate{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_get_object_rate{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - GET", + "range": true, + "refId": "A" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_put_object_rate{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_put_object_rate{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - PUT", + "range": true, + "refId": "B" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_head_object_rate{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_head_object_rate{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - HEAD", + "range": true, + "refId": "C" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_delete_object_rate{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_delete_object_rate{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - DELETE", + "range": true, + "refId": "D" + } + ], + "title": "Top $TopResources SVMs by Rate", + "transformations": [], + "type": "timeseries" + }, + { + "datasource": "${DS_PROMETHEUS}", + "description": "Average latency for GET/PUT/HEAD/DELETE SVM operations.", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "Latency", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 10, + "gradientMode": "opacity", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": true, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "µs", + "unitScale": true + }, + "overrides": [] + }, + "gridPos": { + "h": 11, + "w": 12, + "x": 0, + "y": 26 + }, + "id": 130, + "options": { + "legend": { + "calcs": [ + "mean", + "lastNotNull", + "max" + ], + "displayMode": "table", + "placement": "bottom", + "showLegend": true, + "sortBy": "Last *", + "sortDesc": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "pluginVersion": "8.1.8", + "targets": [ + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_get_object_latency{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_get_object_latency{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - GET", + "range": true, + "refId": "A" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_put_object_latency{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_put_object_latency{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - PUT", + "range": true, + "refId": "B" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_head_object_latency{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_head_object_latency{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - HEAD", + "range": true, + "refId": "C" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_delete_object_latency{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_delete_object_latency{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - DELETE", + "range": true, + "refId": "D" + } + ], + "title": "Top $TopResources SVMs by Latency", + "transformations": [], + "type": "timeseries" + }, + { + "datasource": "${DS_PROMETHEUS}", + "description": "Rate of GET/PUT SVM data transfers per second.", + "fieldConfig": { + "defaults": { + "color": { + "mode": "palette-classic" + }, + "custom": { + "axisBorderShow": false, + "axisCenteredZero": false, + "axisColorMode": "text", + "axisLabel": "Data Transfer", + "axisPlacement": "auto", + "barAlignment": 0, + "drawStyle": "line", + "fillOpacity": 10, + "gradientMode": "opacity", + "hideFrom": { + "legend": false, + "tooltip": false, + "viz": false + }, + "insertNulls": false, + "lineInterpolation": "linear", + "lineWidth": 1, + "pointSize": 5, + "scaleDistribution": { + "type": "linear" + }, + "showPoints": "auto", + "spanNulls": true, + "stacking": { + "group": "A", + "mode": "none" + }, + "thresholdsStyle": { + "mode": "off" + } + }, + "decimals": 2, + "mappings": [], + "thresholds": { + "mode": "absolute", + "steps": [ + { + "color": "green", + "value": null + } + ] + }, + "unit": "Bps", + "unitScale": true + }, + "overrides": [] + }, + "gridPos": { + "h": 11, + "w": 12, + "x": 12, + "y": 26 + }, + "id": 131, + "options": { + "legend": { + "calcs": [ + "mean", + "lastNotNull", + "max" + ], + "displayMode": "table", + "placement": "bottom", + "showLegend": true, + "sortBy": "Last *", + "sortDesc": true + }, + "tooltip": { + "mode": "single", + "sort": "none" + } + }, + "pluginVersion": "8.1.8", + "targets": [ + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_get_data{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_get_data{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - GET", + "range": true, + "refId": "A" + }, + { + "editorMode": "code", + "exemplar": false, + "expr": "avg by(datacenter,cluster,svm)(ontaps3_svm_put_data{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}) and topk($TopResources, avg by(datacenter,cluster,svm) (avg_over_time(ontaps3_svm_put_data{datacenter=~\"$Datacenter\",cluster=~\"$Cluster\",svm=~\"$SVM\"}[3h])))", + "hide": false, + "interval": "", + "legendFormat": "{{svm}} - PUT", + "range": true, + "refId": "B" + } + ], + "title": "Top $TopResources SVMs by Data Transfer", + "transformations": [], + "type": "timeseries" + } + ], + "title": "S3 Object Storage SVM", + "type": "row" } ], "refresh": "", @@ -1157,5 +1897,5 @@ "timezone": "", "title": "ONTAP: S3 Object Storage", "uid": "cdot-s3-object-storage", - "version": 2 + "version": 3 } diff --git a/integration/test/dashboard_json_test.go b/integration/test/dashboard_json_test.go index 2b41ade0f..6698fbbbc 100644 --- a/integration/test/dashboard_json_test.go +++ b/integration/test/dashboard_json_test.go @@ -105,6 +105,7 @@ var excludeCounters = []string{ "ndmp_session_", "metrocluster_check_", "volume_top_clients_", + "ontaps3_svm_", } var flakyCounters = []string{ diff --git a/pkg/matrix/matrix.go b/pkg/matrix/matrix.go index 39aabcec4..3e0e30a55 100644 --- a/pkg/matrix/matrix.go +++ b/pkg/matrix/matrix.go @@ -473,7 +473,7 @@ func (m *Matrix) DivideWithThreshold(metricKey string, baseKey string, threshold case metric.record[i] && sRecord[i]: minimumBase := tValues[i] * x metricName := metric.GetName() - if metricName == "optimal_point_latency" || metricName == "scan_latency" { + if metricName == "optimal_point_latency" || metricName == "scan_latency" || m.Object == "ontaps3_svm" { // An exception is made for these counters because the base counter always has a few IOPS minimumBase = 0 }