WIP: export metrics in the md format #946
7 changed files with 154 additions and 21 deletions
4
Makefile
4
Makefile
|
@ -94,6 +94,10 @@ export-metrics: dep
|
||||||
CGO_ENABLED=0 \
|
CGO_ENABLED=0 \
|
||||||
go build -v -trimpath -o bin/export-metrics ./scripts/export-metrics
|
go build -v -trimpath -o bin/export-metrics ./scripts/export-metrics
|
||||||
|
|
||||||
|
docs: export-metrics
|
||||||
|
@./bin/export-metrics -node ./docs/node-metrics.md -format md
|
||||||
|
@./bin/export-metrics -ir ./docs/ir-metrics.md -format md
|
||||||
|
|
||||||
# Regenerate proto files:
|
# Regenerate proto files:
|
||||||
protoc:
|
protoc:
|
||||||
@if [ ! -d "$(PROTOC_DIR)" ] || [ ! -d "$(PROTOC_GEN_GO_DIR)" ] || [ ! -d "$(PROTOGEN_FROSTFS_DIR)" ]; then \
|
@if [ ! -d "$(PROTOC_DIR)" ] || [ ! -d "$(PROTOC_GEN_GO_DIR)" ] || [ ! -d "$(PROTOGEN_FROSTFS_DIR)" ]; then \
|
||||||
|
|
8
docs/ir-metrics.md
Normal file
8
docs/ir-metrics.md
Normal file
|
@ -0,0 +1,8 @@
|
||||||
|
# FrostFS Node prometheus metrics
|
||||||
|
Name | Description | Constant labels | Variable labels
|
||||||
|
-- | -- | -- | --
|
||||||
|
`frostfs_ir_ir_epoch` | Current epoch as seen by inner-ring node. | |
|
||||||
|
`frostfs_ir_ir_event_duration_seconds` | Duration of processing of inner-ring events | | `type`<br>`success`
|
||||||
|
`frostfs_ir_ir_health` | Current inner-ring node state. | |
|
||||||
|
`frostfs_ir_logger_entry_count` | Total log entries emitted or dropped by severity level | | `level`<br>`dropped`
|
||||||
|
`frostfs_ir_morphcache_request_duration_seconds` | Morph cache request process duration | | `success`<br>`method`
|
59
docs/node-metrics.md
Normal file
59
docs/node-metrics.md
Normal file
|
@ -0,0 +1,59 @@
|
||||||
|
# FrostFS Node prometheus metrics
|
||||||
|
Name | Description | Constant labels | Variable labels
|
||||||
|
-- | -- | -- | --
|
||||||
|
`frostfs_node_blobovnicza_tree_get_bytes` | Accumulated payload size read from Blobovnicza tree | | `shard_id`<br>`path`
|
||||||
|
`frostfs_node_blobovnicza_tree_mode` | Blobovnicza tree mode | | `shard_id`<br>`path`<br>`mode`
|
||||||
|
`frostfs_node_blobovnicza_tree_object_move_duration_seconds` | Accumulated Blobovnicza tree object move duration | | `shard_id`<br>`path`
|
||||||
|
`frostfs_node_blobovnicza_tree_open_blobovnicza_count` | Count of opened blobovniczas of Blobovnicza tree | | `shard_id`<br>`path`
|
||||||
|
`frostfs_node_blobovnicza_tree_open_blobovnicza_items_total` | Count of items in opened blobovniczas of Blobovnicza tree | | `shard_id`<br>`path`
|
||||||
|
`frostfs_node_blobovnicza_tree_open_blobovnicza_size_bytes` | Size of opened blobovniczas of Blobovnicza tree | | `shard_id`<br>`path`
|
||||||
|
`frostfs_node_blobovnicza_tree_put_bytes` | Accumulated payload size written to Blobovnicza tree | | `shard_id`<br>`path`
|
||||||
|
`frostfs_node_blobovnicza_tree_rebuild_complete_percent` | Percent of rebuild completeness | | `shard_id`<br>`path`
|
||||||
|
`frostfs_node_blobovnicza_tree_rebuild_status` | Blobovnicza tree rebuild status | | `shard_id`<br>`path`<br>`mode`
|
||||||
|
`frostfs_node_blobovnicza_tree_request_duration_seconds` | Accumulated Blobovnicza tree request process duration | | `shard_id`<br>`path`<br>`success`<br>`method`<br>`with_storage_id`
|
||||||
|
`frostfs_node_blobstore_get_bytes` | Accumulated payload size read from Blobstore | | `shard_id`
|
||||||
|
`frostfs_node_blobstore_mode` | Blobstore mode value | | `shard_id`<br>`mode`
|
||||||
|
`frostfs_node_blobstore_put_bytes` | Accumulated payload size written to Blobstore | | `shard_id`
|
||||||
|
`frostfs_node_blobstore_request_duration_seconds` | Accumulated Blobstore request process duration | | `shard_id`<br>`success`<br>`method`<br>`with_storage_id`
|
||||||
|
`frostfs_node_engine_container_objects_total` | Count of objects for each container | | `shard_id`<br>`cid`<br>`type`
|
||||||
|
`frostfs_node_engine_container_size_bytes` | Accumulated size of all objects in a container | | `cid`
|
||||||
|
`frostfs_node_engine_errors_total` | Shard's error counter | | `shard_id`
|
||||||
|
`frostfs_node_engine_mode_info` | Shard mode | | `shard_id`<br>`mode`
|
||||||
|
`frostfs_node_engine_objects_total` | Objects counters per shards. DEPRECATED: Will be deleted in next releasese, use frostfs_node_engine_container_objects_total metric. | | `shard_id`<br>`type`
|
||||||
|
`frostfs_node_engine_payload_size_bytes` | Accumulated size of all objects in a shard | | `shard_id`
|
||||||
|
`frostfs_node_engine_request_duration_seconds` | Duration of Engine requests | | `method`
|
||||||
|
`frostfs_node_fstree_get_bytes` | Accumulated payload size read from FSTree | | `shard_id`<br>`path`
|
||||||
|
`frostfs_node_fstree_mode` | FSTree mode value | | `shard_id`<br>`path`<br>`mode`
|
||||||
|
`frostfs_node_fstree_put_bytes` | Accumulated payload size written to FSTree | | `shard_id`<br>`path`
|
||||||
|
`frostfs_node_fstree_request_duration_seconds` | Accumulated FSTree request process duration | | `shard_id`<br>`success`<br>`path`<br>`method`
|
||||||
|
`frostfs_node_garbage_collector_delete_duration_seconds` | The total time of GC runs to delete objects from disk | | `shard_id`<br>`success`
|
||||||
|
`frostfs_node_garbage_collector_deleted_objects_total` | Total count of objects GC deleted or failed to delete from disk | | `shard_id`<br>`status`
|
||||||
|
`frostfs_node_garbage_collector_marked_for_removal_objects_total` | Total count of expired objects GC marked to remove | | `shard_id`<br>`object_type`
|
||||||
|
`frostfs_node_garbage_collector_marking_duration_seconds` | The total time of GC runs to mark expired objects as removed | | `shard_id`<br>`success`<br>`object_type`
|
||||||
|
`frostfs_node_grpc_server_health` | GRPC Server Endpoint health | | `endpoint`
|
||||||
|
`frostfs_node_ir_epoch` | Current epoch as seen by inner-ring node. | |
|
||||||
|
`frostfs_node_logger_entry_count` | Total log entries emitted or dropped by severity level | | `level`<br>`dropped`
|
||||||
|
`frostfs_node_metabase_mode` | Metabase mode | | `shard_id`<br>`path`<br>`mode`
|
||||||
|
`frostfs_node_metabase_request_duration_seconds` | Accumulated Metabase request process duration | | `shard_id`<br>`success`<br>`path`<br>`method`
|
||||||
|
`frostfs_node_morph_invoke_duration_seconds` | Cummulative duration of contract invocations | | `invoke_type`<br>`contract`<br>`method`<br>`success`
|
||||||
|
`frostfs_node_morph_last_block` | Index of the last received block | |
|
||||||
|
`frostfs_node_morph_notifications_total` | Number of notifications received by notification type | | `notification_type`
|
||||||
|
`frostfs_node_morph_switches_total` | Number of endpoint switches | |
|
||||||
|
`frostfs_node_morphcache_request_duration_seconds` | Morph cache request process duration | | `success`<br>`method`
|
||||||
|
`frostfs_node_object_request_duration_seconds` | Object Service request process duration | | `method`<br>`success`
|
||||||
|
`frostfs_node_object_request_payload_bytes` | Object Service request payload | | `method`
|
||||||
|
`frostfs_node_pilorama_mode` | Pilorama mode | | `shard_id`<br>`mode`
|
||||||
|
`frostfs_node_pilorama_request_duration_seconds` | Accumulated Pilorama request process duration | | `shard_id`<br>`success`<br>`method`
|
||||||
|
`frostfs_node_policer_processed_objects_total` | Total number of objects processed by policer | |
|
||||||
|
`frostfs_node_replicator_in_flight_requests_total` | Number of in-flight requests | |
|
||||||
|
`frostfs_node_replicator_processed_objects_total` | Number of objects processed since the node startup | |
|
||||||
|
`frostfs_node_replicator_total_replicated_payload_size_bytes` | Total size of payloads replicated | |
|
||||||
|
`frostfs_node_state_health` | Current Node state | |
|
||||||
|
`frostfs_node_treeservice_replicate_task_duration_seconds` | Duration of individual replication tasks executed as part of replication loops | | `success`
|
||||||
|
`frostfs_node_treeservice_replicate_wait_duration_seconds` | Duration of overall waiting time for replication loops | | `success`
|
||||||
|
`frostfs_node_treeservice_sync_duration_seconds` | Duration of synchronization operations | | `success`
|
||||||
|
`frostfs_node_writecache_actual_objects_total` | Actual objects count in writecache | | `shard_id`<br>`storage`
|
||||||
|
`frostfs_node_writecache_estimated_size_bytes` | Estimated writecache size | | `shard_id`<br>`storage`
|
||||||
|
`frostfs_node_writecache_mode_info` | Writecache mode value | | `shard_id`<br>`mode`
|
||||||
|
`frostfs_node_writecache_operations_total` | The number of writecache operations processed | | `shard_id`<br>`storage`<br>`success`<br>`operation`
|
||||||
|
`frostfs_node_writecache_request_duration_seconds` | Writecache request process duration | | `shard_id`<br>`success`<br>`storage`<br>`method`
|
28
go.mod
28
go.mod
|
@ -3,27 +3,27 @@ module git.frostfs.info/TrueCloudLab/frostfs-node
|
||||||
go 1.20
|
go 1.20
|
||||||
|
|
||||||
require (
|
require (
|
||||||
git.frostfs.info/TrueCloudLab/frostfs-api-go/v2 v2.16.1-0.20240112150928-72885aae835c
|
git.frostfs.info/TrueCloudLab/frostfs-api-go/v2 v2.16.1-0.20240126091953-d60ce83e4271
|
||||||
git.frostfs.info/TrueCloudLab/frostfs-contract v0.18.1-0.20240115082915-f2a82aa635aa
|
git.frostfs.info/TrueCloudLab/frostfs-contract v0.18.1-0.20240130133737-6e72d0b3b405
|
||||||
git.frostfs.info/TrueCloudLab/frostfs-observability v0.0.0-20231101111734-b3ad3335ff65
|
git.frostfs.info/TrueCloudLab/frostfs-observability v0.0.0-20231101111734-b3ad3335ff65
|
||||||
git.frostfs.info/TrueCloudLab/frostfs-sdk-go v0.0.0-20240117145620-110b7e41706e
|
git.frostfs.info/TrueCloudLab/frostfs-sdk-go v0.0.0-20240126141009-65b4525b3bf0
|
||||||
git.frostfs.info/TrueCloudLab/hrw v1.2.1
|
git.frostfs.info/TrueCloudLab/hrw v1.2.1
|
||||||
git.frostfs.info/TrueCloudLab/policy-engine v0.0.0-20240122104724-06cbfe8691ad
|
git.frostfs.info/TrueCloudLab/policy-engine v0.0.0-20240129064140-8d21ab2d99d9
|
||||||
git.frostfs.info/TrueCloudLab/tzhash v1.8.0
|
git.frostfs.info/TrueCloudLab/tzhash v1.8.0
|
||||||
github.com/cheggaaa/pb v1.0.29
|
github.com/cheggaaa/pb v1.0.29
|
||||||
github.com/chzyer/readline v1.5.1
|
github.com/chzyer/readline v1.5.1
|
||||||
github.com/flynn-archive/go-shlex v0.0.0-20150515145356-3f9db97f8568
|
github.com/flynn-archive/go-shlex v0.0.0-20150515145356-3f9db97f8568
|
||||||
github.com/google/uuid v1.6.0
|
github.com/google/uuid v1.6.0
|
||||||
github.com/hashicorp/golang-lru/v2 v2.0.7
|
github.com/hashicorp/golang-lru/v2 v2.0.7
|
||||||
github.com/klauspost/compress v1.17.4
|
github.com/klauspost/compress v1.17.5
|
||||||
github.com/mitchellh/go-homedir v1.1.0
|
github.com/mitchellh/go-homedir v1.1.0
|
||||||
github.com/mr-tron/base58 v1.2.0
|
github.com/mr-tron/base58 v1.2.0
|
||||||
github.com/multiformats/go-multiaddr v0.12.1
|
github.com/multiformats/go-multiaddr v0.12.2
|
||||||
github.com/nats-io/nats.go v1.32.0
|
github.com/nats-io/nats.go v1.32.0
|
||||||
github.com/nspcc-dev/neo-go v0.105.1
|
github.com/nspcc-dev/neo-go v0.105.1
|
||||||
github.com/olekukonko/tablewriter v0.0.5
|
github.com/olekukonko/tablewriter v0.0.5
|
||||||
github.com/panjf2000/ants/v2 v2.9.0
|
github.com/panjf2000/ants/v2 v2.9.0
|
||||||
github.com/paulmach/orb v0.11.0
|
github.com/paulmach/orb v0.11.1
|
||||||
github.com/prometheus/client_golang v1.18.0
|
github.com/prometheus/client_golang v1.18.0
|
||||||
github.com/spf13/cast v1.6.0
|
github.com/spf13/cast v1.6.0
|
||||||
github.com/spf13/cobra v1.8.0
|
github.com/spf13/cobra v1.8.0
|
||||||
|
@ -58,7 +58,7 @@ require (
|
||||||
github.com/cenkalti/backoff/v4 v4.2.1 // indirect
|
github.com/cenkalti/backoff/v4 v4.2.1 // indirect
|
||||||
github.com/cespare/xxhash/v2 v2.2.0 // indirect
|
github.com/cespare/xxhash/v2 v2.2.0 // indirect
|
||||||
github.com/consensys/bavard v0.1.13 // indirect
|
github.com/consensys/bavard v0.1.13 // indirect
|
||||||
github.com/consensys/gnark-crypto v0.12.2-0.20231222162921-eb75782795d2 // indirect
|
github.com/consensys/gnark-crypto v0.12.2-0.20240124182927-2e4aaaaefdbf // indirect
|
||||||
github.com/cpuguy83/go-md2man/v2 v2.0.3 // indirect
|
github.com/cpuguy83/go-md2man/v2 v2.0.3 // indirect
|
||||||
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect
|
github.com/davecgh/go-spew v1.1.2-0.20180830191138-d8f796af33cc // indirect
|
||||||
github.com/decred/dcrd/dcrec/secp256k1/v4 v4.2.0 // indirect
|
github.com/decred/dcrd/dcrec/secp256k1/v4 v4.2.0 // indirect
|
||||||
|
@ -70,14 +70,16 @@ require (
|
||||||
github.com/gorilla/websocket v1.5.1 // indirect
|
github.com/gorilla/websocket v1.5.1 // indirect
|
||||||
github.com/grpc-ecosystem/go-grpc-middleware/providers/prometheus v1.0.0 // indirect
|
github.com/grpc-ecosystem/go-grpc-middleware/providers/prometheus v1.0.0 // indirect
|
||||||
github.com/grpc-ecosystem/go-grpc-middleware/v2 v2.0.1 // indirect
|
github.com/grpc-ecosystem/go-grpc-middleware/v2 v2.0.1 // indirect
|
||||||
github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.0 // indirect
|
github.com/grpc-ecosystem/grpc-gateway/v2 v2.19.1 // indirect
|
||||||
github.com/hashicorp/golang-lru v1.0.2 // indirect
|
github.com/hashicorp/golang-lru v1.0.2 // indirect
|
||||||
github.com/hashicorp/hcl v1.0.0 // indirect
|
github.com/hashicorp/hcl v1.0.0 // indirect
|
||||||
github.com/holiman/uint256 v1.2.4 // indirect
|
github.com/holiman/uint256 v1.2.4 // indirect
|
||||||
github.com/inconshreveable/mousetrap v1.1.0 // indirect
|
github.com/inconshreveable/mousetrap v1.1.0 // indirect
|
||||||
github.com/ipfs/go-cid v0.4.1 // indirect
|
github.com/ipfs/go-cid v0.4.1 // indirect
|
||||||
|
github.com/josharian/intern v1.0.0 // indirect
|
||||||
github.com/klauspost/cpuid/v2 v2.2.6 // indirect
|
github.com/klauspost/cpuid/v2 v2.2.6 // indirect
|
||||||
github.com/magiconair/properties v1.8.7 // indirect
|
github.com/magiconair/properties v1.8.7 // indirect
|
||||||
|
github.com/mailru/easyjson v0.7.7 // indirect
|
||||||
github.com/mattn/go-runewidth v0.0.15 // indirect
|
github.com/mattn/go-runewidth v0.0.15 // indirect
|
||||||
github.com/minio/sha256-simd v1.0.1 // indirect
|
github.com/minio/sha256-simd v1.0.1 // indirect
|
||||||
github.com/mitchellh/mapstructure v1.5.0 // indirect
|
github.com/mitchellh/mapstructure v1.5.0 // indirect
|
||||||
|
@ -90,14 +92,14 @@ require (
|
||||||
github.com/nats-io/nkeys v0.4.7 // indirect
|
github.com/nats-io/nkeys v0.4.7 // indirect
|
||||||
github.com/nats-io/nuid v1.0.1 // indirect
|
github.com/nats-io/nuid v1.0.1 // indirect
|
||||||
github.com/nspcc-dev/go-ordered-json v0.0.0-20240112074137-296698a162ae // indirect
|
github.com/nspcc-dev/go-ordered-json v0.0.0-20240112074137-296698a162ae // indirect
|
||||||
github.com/nspcc-dev/neo-go/pkg/interop v0.0.0-20240122090917-ef99a7a9e33f // indirect
|
github.com/nspcc-dev/neo-go/pkg/interop v0.0.0-20240127082647-51ac153a7b00 // indirect
|
||||||
github.com/nspcc-dev/rfc6979 v0.2.0 // indirect
|
github.com/nspcc-dev/rfc6979 v0.2.0 // indirect
|
||||||
github.com/pelletier/go-toml/v2 v2.1.1 // indirect
|
github.com/pelletier/go-toml/v2 v2.1.1 // indirect
|
||||||
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect
|
github.com/pmezard/go-difflib v1.0.1-0.20181226105442-5d4384ee4fb2 // indirect
|
||||||
github.com/prometheus/client_model v0.5.0 // indirect
|
github.com/prometheus/client_model v0.5.0 // indirect
|
||||||
github.com/prometheus/common v0.46.0 // indirect
|
github.com/prometheus/common v0.46.0 // indirect
|
||||||
github.com/prometheus/procfs v0.12.0 // indirect
|
github.com/prometheus/procfs v0.12.0 // indirect
|
||||||
github.com/rivo/uniseg v0.4.4 // indirect
|
github.com/rivo/uniseg v0.4.6 // indirect
|
||||||
github.com/russross/blackfriday/v2 v2.1.0 // indirect
|
github.com/russross/blackfriday/v2 v2.1.0 // indirect
|
||||||
github.com/spaolacci/murmur3 v1.1.0 // indirect
|
github.com/spaolacci/murmur3 v1.1.0 // indirect
|
||||||
github.com/spf13/afero v1.11.0 // indirect
|
github.com/spf13/afero v1.11.0 // indirect
|
||||||
|
@ -116,8 +118,8 @@ require (
|
||||||
golang.org/x/net v0.20.0 // indirect
|
golang.org/x/net v0.20.0 // indirect
|
||||||
golang.org/x/sys v0.16.0 // indirect
|
golang.org/x/sys v0.16.0 // indirect
|
||||||
golang.org/x/text v0.14.0 // indirect
|
golang.org/x/text v0.14.0 // indirect
|
||||||
google.golang.org/genproto/googleapis/api v0.0.0-20240123012728-ef4313101c80 // indirect
|
google.golang.org/genproto/googleapis/api v0.0.0-20240125205218-1f4bbc51befe // indirect
|
||||||
google.golang.org/genproto/googleapis/rpc v0.0.0-20240123012728-ef4313101c80 // indirect
|
google.golang.org/genproto/googleapis/rpc v0.0.0-20240125205218-1f4bbc51befe // indirect
|
||||||
gopkg.in/ini.v1 v1.67.0 // indirect
|
gopkg.in/ini.v1 v1.67.0 // indirect
|
||||||
lukechampine.com/blake3 v1.2.1 // indirect
|
lukechampine.com/blake3 v1.2.1 // indirect
|
||||||
rsc.io/tmplfunc v0.0.3 // indirect
|
rsc.io/tmplfunc v0.0.3 // indirect
|
||||||
|
|
BIN
go.sum
BIN
go.sum
Binary file not shown.
27
scripts/changelog-updates/script.sh
Executable file
27
scripts/changelog-updates/script.sh
Executable file
|
@ -0,0 +1,27 @@
|
||||||
|
#!/bin/bash
|
||||||
|
|
||||||
|
FROM="$1"
|
||||||
|
OLD_FILE="$(mktemp)"
|
||||||
|
NEW_FILE="$(mktemp)"
|
||||||
|
|
||||||
|
declare -A old
|
||||||
|
git show "$FROM:go.mod" \
|
||||||
|
| sed -n -e '6 { /require/ b x; :x { p; n; /)/ q; b x} }' > "$OLD_FILE"
|
||||||
|
while IFS= read -r line; do
|
||||||
|
x=( $line )
|
||||||
|
old[${x[0]}]=${x[1]}
|
||||||
|
done < "$OLD_FILE"
|
||||||
|
|
||||||
|
declare -A new
|
||||||
|
cat go.mod \
|
||||||
|
| sed -n -e '6 { /require/ b x; :x { p; n; /)/ q; b x} }' > "$NEW_FILE"
|
||||||
|
while IFS= read -r line; do
|
||||||
|
x=( $line )
|
||||||
|
new[${x[0]}]=${x[1]}
|
||||||
|
done < "$NEW_FILE"
|
||||||
|
|
||||||
|
for index in ${!new[@]}; do
|
||||||
|
if [[ -n $old[$index] ]] && [[ ${old[$index]} != ${new[$index]} ]]; then
|
||||||
|
echo "- \`$index\` from ${old[$index]} to ${new[$index]}"
|
||||||
|
fi
|
||||||
|
done
|
|
@ -1,10 +1,13 @@
|
||||||
package main
|
package main
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"bytes"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
"flag"
|
"flag"
|
||||||
"fmt"
|
"fmt"
|
||||||
"os"
|
"os"
|
||||||
|
"sort"
|
||||||
|
"strings"
|
||||||
|
|
||||||
local_metrics "git.frostfs.info/TrueCloudLab/frostfs-node/pkg/metrics"
|
local_metrics "git.frostfs.info/TrueCloudLab/frostfs-node/pkg/metrics"
|
||||||
"git.frostfs.info/TrueCloudLab/frostfs-observability/metrics"
|
"git.frostfs.info/TrueCloudLab/frostfs-observability/metrics"
|
||||||
|
@ -13,6 +16,7 @@ import (
|
||||||
var (
|
var (
|
||||||
node = flag.String("node", "", "File to export storage node metrics to.")
|
node = flag.String("node", "", "File to export storage node metrics to.")
|
||||||
ir = flag.String("ir", "", "File to export innerring node metrics to.")
|
ir = flag.String("ir", "", "File to export innerring node metrics to.")
|
||||||
|
format = flag.String("format", "json", "Format to export in. Supported formats: json (default), md.")
|
||||||
)
|
)
|
||||||
|
|
||||||
func main() {
|
func main() {
|
||||||
|
@ -37,13 +41,42 @@ func main() {
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
|
|
||||||
ds := metrics.DescribeAll()
|
var data []byte
|
||||||
|
var err error
|
||||||
|
|
||||||
data, err := json.Marshal(ds)
|
ds := metrics.DescribeAll()
|
||||||
|
switch *format {
|
||||||
|
case "md":
|
||||||
|
sort.Slice(ds, func(i, j int) bool {
|
||||||
|
return ds[i].Name < ds[j].Name
|
||||||
|
})
|
||||||
|
b := bytes.NewBuffer(nil)
|
||||||
|
b.WriteString("# FrostFS Node prometheus metrics\n")
|
||||||
|
b.WriteString("Name | Description | Constant labels | Variable labels\n")
|
||||||
|
b.WriteString("-- | -- | -- | --\n")
|
||||||
|
for i := range ds {
|
||||||
|
constLabels := make([]string, 0, len(ds[i].ConstantLabels))
|
||||||
|
for key, value := range ds[i].ConstantLabels {
|
||||||
|
constLabels = append(constLabels, fmt.Sprintf("`%s=%s`", key, value))
|
||||||
|
}
|
||||||
|
|
||||||
|
// FIXME omg do it better
|
||||||
|
ss := make([]string, len(ds[i].VariableLabels))
|
||||||
|
copy(ss, ds[i].VariableLabels)
|
||||||
|
for i := range ss {
|
||||||
|
ss[i] = fmt.Sprintf("`%s`", ss[i])
|
||||||
|
}
|
||||||
|
varLabels := strings.Join(ss, "<br>")
|
||||||
|
fmt.Fprintf(b, "`%s` | %s | %s | %s\n", ds[i].Name, ds[i].Help, strings.Join(constLabels, "<br>"), varLabels)
|
||||||
|
}
|
||||||
|
data = b.Bytes()
|
||||||
|
default:
|
||||||
|
data, err = json.Marshal(ds)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Fprintf(os.Stderr, "Could not parse marshal: %v\n", err)
|
fmt.Fprintf(os.Stderr, "Could not parse marshal: %v\n", err)
|
||||||
os.Exit(1)
|
os.Exit(1)
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if err := os.WriteFile(filename, data, 0o644); err != nil {
|
if err := os.WriteFile(filename, data, 0o644); err != nil {
|
||||||
fmt.Fprintf(os.Stderr, "Could write to file: %v\n", err)
|
fmt.Fprintf(os.Stderr, "Could write to file: %v\n", err)
|
||||||
|
|
Loading…
Reference in a new issue