mirror of
https://github.com/0glabs/0g-chain.git
synced 2024-12-27 16:55:21 +00:00
* feat(x/metrics): add module for emiting custom chain metrics (#1668)
* initialize x/metrics with metrics collection
* include global labels in x/metrics metrics
* add x/metrics spec
* add x/metrics test coverage
* update changelog
(cherry picked from commit 9a0aed7626
)
# Conflicts:
# CHANGELOG.md
* fix changlog conflicts
---------
Co-authored-by: Robert Pirtle <Astropirtle@gmail.com>
This commit is contained in:
parent
48c845b941
commit
6c565343f7
@ -36,6 +36,11 @@ Ref: https://keepachangelog.com/en/1.0.0/
|
|||||||
|
|
||||||
## [Unreleased]
|
## [Unreleased]
|
||||||
|
|
||||||
|
### Features
|
||||||
|
- (metrics) [#1668] Adds non-state breaking x/metrics module for custom telemetry.
|
||||||
|
|
||||||
|
## [v0.24.0](https://github.com/Kava-Labs/kava/releases/tag/v0.24.0)
|
||||||
|
|
||||||
### Features
|
### Features
|
||||||
- (evmutil) [#1590] & [#1596] Add allow list param of sdk native denoms that can be transferred to evm
|
- (evmutil) [#1590] & [#1596] Add allow list param of sdk native denoms that can be transferred to evm
|
||||||
- (evmutil) [#1591] & [#1596] Configure module to support deploying ERC20KavaWrappedCosmosCoin contracts
|
- (evmutil) [#1591] & [#1596] Configure module to support deploying ERC20KavaWrappedCosmosCoin contracts
|
||||||
@ -268,6 +273,7 @@ the [changelog](https://github.com/cosmos/cosmos-sdk/blob/v0.38.4/CHANGELOG.md).
|
|||||||
- [#257](https://github.com/Kava-Labs/kava/pulls/257) Include scripts to run
|
- [#257](https://github.com/Kava-Labs/kava/pulls/257) Include scripts to run
|
||||||
large-scale simulations remotely using aws-batch
|
large-scale simulations remotely using aws-batch
|
||||||
|
|
||||||
|
[#1668]: https://github.com/Kava-Labs/kava/pull/1668
|
||||||
[#1624]: https://github.com/Kava-Labs/kava/pull/1624
|
[#1624]: https://github.com/Kava-Labs/kava/pull/1624
|
||||||
[#1622]: https://github.com/Kava-Labs/kava/pull/1622
|
[#1622]: https://github.com/Kava-Labs/kava/pull/1622
|
||||||
[#1614]: https://github.com/Kava-Labs/kava/pull/1614
|
[#1614]: https://github.com/Kava-Labs/kava/pull/1614
|
||||||
|
@ -139,6 +139,8 @@ import (
|
|||||||
"github.com/kava-labs/kava/x/liquid"
|
"github.com/kava-labs/kava/x/liquid"
|
||||||
liquidkeeper "github.com/kava-labs/kava/x/liquid/keeper"
|
liquidkeeper "github.com/kava-labs/kava/x/liquid/keeper"
|
||||||
liquidtypes "github.com/kava-labs/kava/x/liquid/types"
|
liquidtypes "github.com/kava-labs/kava/x/liquid/types"
|
||||||
|
metrics "github.com/kava-labs/kava/x/metrics"
|
||||||
|
metricstypes "github.com/kava-labs/kava/x/metrics/types"
|
||||||
pricefeed "github.com/kava-labs/kava/x/pricefeed"
|
pricefeed "github.com/kava-labs/kava/x/pricefeed"
|
||||||
pricefeedkeeper "github.com/kava-labs/kava/x/pricefeed/keeper"
|
pricefeedkeeper "github.com/kava-labs/kava/x/pricefeed/keeper"
|
||||||
pricefeedtypes "github.com/kava-labs/kava/x/pricefeed/types"
|
pricefeedtypes "github.com/kava-labs/kava/x/pricefeed/types"
|
||||||
@ -216,6 +218,7 @@ var (
|
|||||||
router.AppModuleBasic{},
|
router.AppModuleBasic{},
|
||||||
mint.AppModuleBasic{},
|
mint.AppModuleBasic{},
|
||||||
community.AppModuleBasic{},
|
community.AppModuleBasic{},
|
||||||
|
metrics.AppModuleBasic{},
|
||||||
)
|
)
|
||||||
|
|
||||||
// module account permissions
|
// module account permissions
|
||||||
@ -261,6 +264,7 @@ type Options struct {
|
|||||||
MempoolAuthAddresses []sdk.AccAddress
|
MempoolAuthAddresses []sdk.AccAddress
|
||||||
EVMTrace string
|
EVMTrace string
|
||||||
EVMMaxGasWanted uint64
|
EVMMaxGasWanted uint64
|
||||||
|
TelemetryOptions metricstypes.TelemetryOptions
|
||||||
}
|
}
|
||||||
|
|
||||||
// DefaultOptions is a sensible default Options value.
|
// DefaultOptions is a sensible default Options value.
|
||||||
@ -790,10 +794,12 @@ func NewApp(
|
|||||||
// nil InflationCalculationFn, use SDK's default inflation function
|
// nil InflationCalculationFn, use SDK's default inflation function
|
||||||
mint.NewAppModule(appCodec, app.mintKeeper, app.accountKeeper, nil),
|
mint.NewAppModule(appCodec, app.mintKeeper, app.accountKeeper, nil),
|
||||||
community.NewAppModule(app.communityKeeper, app.accountKeeper),
|
community.NewAppModule(app.communityKeeper, app.accountKeeper),
|
||||||
|
metrics.NewAppModule(options.TelemetryOptions),
|
||||||
)
|
)
|
||||||
|
|
||||||
// Warning: Some begin blockers must run before others. Ensure the dependencies are understood before modifying this list.
|
// Warning: Some begin blockers must run before others. Ensure the dependencies are understood before modifying this list.
|
||||||
app.mm.SetOrderBeginBlockers(
|
app.mm.SetOrderBeginBlockers(
|
||||||
|
metricstypes.ModuleName,
|
||||||
// Upgrade begin blocker runs migrations on the first block after an upgrade. It should run before any other module.
|
// Upgrade begin blocker runs migrations on the first block after an upgrade. It should run before any other module.
|
||||||
upgradetypes.ModuleName,
|
upgradetypes.ModuleName,
|
||||||
// Capability begin blocker runs non state changing initialization.
|
// Capability begin blocker runs non state changing initialization.
|
||||||
@ -882,6 +888,7 @@ func NewApp(
|
|||||||
routertypes.ModuleName,
|
routertypes.ModuleName,
|
||||||
minttypes.ModuleName,
|
minttypes.ModuleName,
|
||||||
communitytypes.ModuleName,
|
communitytypes.ModuleName,
|
||||||
|
metricstypes.ModuleName,
|
||||||
)
|
)
|
||||||
|
|
||||||
// Warning: Some init genesis methods must run before others. Ensure the dependencies are understood before modifying this list
|
// Warning: Some init genesis methods must run before others. Ensure the dependencies are understood before modifying this list
|
||||||
@ -923,6 +930,7 @@ func NewApp(
|
|||||||
validatorvestingtypes.ModuleName,
|
validatorvestingtypes.ModuleName,
|
||||||
liquidtypes.ModuleName,
|
liquidtypes.ModuleName,
|
||||||
routertypes.ModuleName,
|
routertypes.ModuleName,
|
||||||
|
metricstypes.ModuleName,
|
||||||
)
|
)
|
||||||
|
|
||||||
app.mm.RegisterInvariants(&app.crisisKeeper)
|
app.mm.RegisterInvariants(&app.crisisKeeper)
|
||||||
|
@ -24,6 +24,7 @@ import (
|
|||||||
|
|
||||||
"github.com/kava-labs/kava/app"
|
"github.com/kava-labs/kava/app"
|
||||||
"github.com/kava-labs/kava/app/params"
|
"github.com/kava-labs/kava/app/params"
|
||||||
|
metricstypes "github.com/kava-labs/kava/x/metrics/types"
|
||||||
)
|
)
|
||||||
|
|
||||||
const (
|
const (
|
||||||
@ -99,6 +100,7 @@ func (ac appCreator) newApp(
|
|||||||
MempoolAuthAddresses: mempoolAuthAddresses,
|
MempoolAuthAddresses: mempoolAuthAddresses,
|
||||||
EVMTrace: cast.ToString(appOpts.Get(ethermintflags.EVMTracer)),
|
EVMTrace: cast.ToString(appOpts.Get(ethermintflags.EVMTracer)),
|
||||||
EVMMaxGasWanted: cast.ToUint64(appOpts.Get(ethermintflags.EVMMaxTxGasWanted)),
|
EVMMaxGasWanted: cast.ToUint64(appOpts.Get(ethermintflags.EVMMaxTxGasWanted)),
|
||||||
|
TelemetryOptions: metricstypes.TelemetryOptionsFromAppOpts(appOpts),
|
||||||
},
|
},
|
||||||
baseapp.SetPruning(pruningOpts),
|
baseapp.SetPruning(pruningOpts),
|
||||||
baseapp.SetMinGasPrices(strings.Replace(cast.ToString(appOpts.Get(server.FlagMinGasPrices)), ";", ",", -1)),
|
baseapp.SetMinGasPrices(strings.Replace(cast.ToString(appOpts.Get(server.FlagMinGasPrices)), ";", ",", -1)),
|
||||||
|
4
go.mod
4
go.mod
@ -12,11 +12,13 @@ require (
|
|||||||
github.com/cosmos/ibc-go/v6 v6.1.1
|
github.com/cosmos/ibc-go/v6 v6.1.1
|
||||||
github.com/ethereum/go-ethereum v1.10.26
|
github.com/ethereum/go-ethereum v1.10.26
|
||||||
github.com/evmos/ethermint v0.21.0
|
github.com/evmos/ethermint v0.21.0
|
||||||
|
github.com/go-kit/kit v0.12.0
|
||||||
github.com/gogo/protobuf v1.3.3
|
github.com/gogo/protobuf v1.3.3
|
||||||
github.com/golang/protobuf v1.5.3
|
github.com/golang/protobuf v1.5.3
|
||||||
github.com/gorilla/mux v1.8.0
|
github.com/gorilla/mux v1.8.0
|
||||||
github.com/grpc-ecosystem/grpc-gateway v1.16.0
|
github.com/grpc-ecosystem/grpc-gateway v1.16.0
|
||||||
github.com/pelletier/go-toml/v2 v2.0.6
|
github.com/pelletier/go-toml/v2 v2.0.6
|
||||||
|
github.com/prometheus/client_golang v1.14.0
|
||||||
github.com/spf13/cast v1.5.0
|
github.com/spf13/cast v1.5.0
|
||||||
github.com/spf13/cobra v1.6.1
|
github.com/spf13/cobra v1.6.1
|
||||||
github.com/spf13/viper v1.15.0
|
github.com/spf13/viper v1.15.0
|
||||||
@ -84,7 +86,6 @@ require (
|
|||||||
github.com/fsnotify/fsnotify v1.6.0 // indirect
|
github.com/fsnotify/fsnotify v1.6.0 // indirect
|
||||||
github.com/gballet/go-libpcsclite v0.0.0-20190607065134-2772fd86a8ff // indirect
|
github.com/gballet/go-libpcsclite v0.0.0-20190607065134-2772fd86a8ff // indirect
|
||||||
github.com/gin-gonic/gin v1.8.1 // indirect
|
github.com/gin-gonic/gin v1.8.1 // indirect
|
||||||
github.com/go-kit/kit v0.12.0 // indirect
|
|
||||||
github.com/go-kit/log v0.2.1 // indirect
|
github.com/go-kit/log v0.2.1 // indirect
|
||||||
github.com/go-logfmt/logfmt v0.5.1 // indirect
|
github.com/go-logfmt/logfmt v0.5.1 // indirect
|
||||||
github.com/go-ole/go-ole v1.2.6 // indirect
|
github.com/go-ole/go-ole v1.2.6 // indirect
|
||||||
@ -148,7 +149,6 @@ require (
|
|||||||
github.com/petermattis/goid v0.0.0-20180202154549-b0b1615b78e5 // indirect
|
github.com/petermattis/goid v0.0.0-20180202154549-b0b1615b78e5 // indirect
|
||||||
github.com/pkg/errors v0.9.1 // indirect
|
github.com/pkg/errors v0.9.1 // indirect
|
||||||
github.com/pmezard/go-difflib v1.0.0 // indirect
|
github.com/pmezard/go-difflib v1.0.0 // indirect
|
||||||
github.com/prometheus/client_golang v1.14.0 // indirect
|
|
||||||
github.com/prometheus/client_model v0.3.0 // indirect
|
github.com/prometheus/client_model v0.3.0 // indirect
|
||||||
github.com/prometheus/common v0.40.0 // indirect
|
github.com/prometheus/common v0.40.0 // indirect
|
||||||
github.com/prometheus/procfs v0.9.0 // indirect
|
github.com/prometheus/procfs v0.9.0 // indirect
|
||||||
|
12
x/metrics/abci.go
Normal file
12
x/metrics/abci.go
Normal file
@ -0,0 +1,12 @@
|
|||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/kava-labs/kava/x/metrics/types"
|
||||||
|
|
||||||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
// BeginBlocker publishes metrics at the start of each block.
|
||||||
|
func BeginBlocker(ctx sdk.Context, metrics *types.Metrics) {
|
||||||
|
metrics.LatestBlockHeight.Set(float64(ctx.BlockHeight()))
|
||||||
|
}
|
45
x/metrics/abci_test.go
Normal file
45
x/metrics/abci_test.go
Normal file
@ -0,0 +1,45 @@
|
|||||||
|
package metrics_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
kitmetrics "github.com/go-kit/kit/metrics"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
|
||||||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
tmproto "github.com/tendermint/tendermint/proto/tendermint/types"
|
||||||
|
|
||||||
|
"github.com/kava-labs/kava/app"
|
||||||
|
"github.com/kava-labs/kava/x/metrics"
|
||||||
|
"github.com/kava-labs/kava/x/metrics/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
type MockGauge struct {
|
||||||
|
value float64
|
||||||
|
}
|
||||||
|
|
||||||
|
func (mg *MockGauge) With(labelValues ...string) kitmetrics.Gauge { return mg }
|
||||||
|
func (mg *MockGauge) Set(value float64) { mg.value = value }
|
||||||
|
func (*MockGauge) Add(_ float64) {}
|
||||||
|
|
||||||
|
func ctxWithHeight(height int64) sdk.Context {
|
||||||
|
tApp := app.NewTestApp()
|
||||||
|
tApp.InitializeFromGenesisStates()
|
||||||
|
return tApp.NewContext(false, tmproto.Header{Height: height})
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestBeginBlockEmitsLatestHeight(t *testing.T) {
|
||||||
|
gauge := MockGauge{}
|
||||||
|
myMetrics := &types.Metrics{
|
||||||
|
LatestBlockHeight: &gauge,
|
||||||
|
}
|
||||||
|
|
||||||
|
metrics.BeginBlocker(ctxWithHeight(1), myMetrics)
|
||||||
|
require.EqualValues(t, 1, gauge.value)
|
||||||
|
|
||||||
|
metrics.BeginBlocker(ctxWithHeight(100), myMetrics)
|
||||||
|
require.EqualValues(t, 100, gauge.value)
|
||||||
|
|
||||||
|
metrics.BeginBlocker(ctxWithHeight(17e6), myMetrics)
|
||||||
|
require.EqualValues(t, 17e6, gauge.value)
|
||||||
|
}
|
125
x/metrics/module.go
Normal file
125
x/metrics/module.go
Normal file
@ -0,0 +1,125 @@
|
|||||||
|
package metrics
|
||||||
|
|
||||||
|
import (
|
||||||
|
"encoding/json"
|
||||||
|
|
||||||
|
"github.com/grpc-ecosystem/grpc-gateway/runtime"
|
||||||
|
"github.com/spf13/cobra"
|
||||||
|
|
||||||
|
"github.com/cosmos/cosmos-sdk/client"
|
||||||
|
"github.com/cosmos/cosmos-sdk/codec"
|
||||||
|
codectypes "github.com/cosmos/cosmos-sdk/codec/types"
|
||||||
|
sdk "github.com/cosmos/cosmos-sdk/types"
|
||||||
|
"github.com/cosmos/cosmos-sdk/types/module"
|
||||||
|
|
||||||
|
abci "github.com/tendermint/tendermint/abci/types"
|
||||||
|
|
||||||
|
"github.com/kava-labs/kava/x/metrics/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
var (
|
||||||
|
_ module.AppModule = AppModule{}
|
||||||
|
_ module.AppModuleBasic = AppModuleBasic{}
|
||||||
|
)
|
||||||
|
|
||||||
|
// AppModuleBasic app module basics object
|
||||||
|
type AppModuleBasic struct{}
|
||||||
|
|
||||||
|
// Name returns the module name
|
||||||
|
func (AppModuleBasic) Name() string {
|
||||||
|
return types.ModuleName
|
||||||
|
}
|
||||||
|
|
||||||
|
// RegisterLegacyAminoCodec register module codec
|
||||||
|
// Deprecated: unused but necessary to fulfill AppModuleBasic interface
|
||||||
|
func (AppModuleBasic) RegisterLegacyAminoCodec(cdc *codec.LegacyAmino) {}
|
||||||
|
|
||||||
|
// DefaultGenesis default genesis state
|
||||||
|
func (AppModuleBasic) DefaultGenesis(_ codec.JSONCodec) json.RawMessage {
|
||||||
|
return []byte("{}")
|
||||||
|
}
|
||||||
|
|
||||||
|
// ValidateGenesis module validate genesis
|
||||||
|
func (AppModuleBasic) ValidateGenesis(_ codec.JSONCodec, _ client.TxEncodingConfig, _ json.RawMessage) error {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// RegisterInterfaces implements InterfaceModule.RegisterInterfaces
|
||||||
|
func (a AppModuleBasic) RegisterInterfaces(registry codectypes.InterfaceRegistry) {}
|
||||||
|
|
||||||
|
// RegisterGRPCGatewayRoutes registers the gRPC Gateway routes for the module.
|
||||||
|
func (a AppModuleBasic) RegisterGRPCGatewayRoutes(clientCtx client.Context, mux *runtime.ServeMux) {}
|
||||||
|
|
||||||
|
// GetTxCmd returns the root tx command for the module.
|
||||||
|
func (AppModuleBasic) GetTxCmd() *cobra.Command {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// GetQueryCmd returns no root query command for the module.
|
||||||
|
func (AppModuleBasic) GetQueryCmd() *cobra.Command {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
//____________________________________________________________________________
|
||||||
|
|
||||||
|
// AppModule app module type
|
||||||
|
type AppModule struct {
|
||||||
|
AppModuleBasic
|
||||||
|
metrics *types.Metrics
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewAppModule creates a new AppModule object
|
||||||
|
func NewAppModule(telemetryOpts types.TelemetryOptions) AppModule {
|
||||||
|
return AppModule{
|
||||||
|
AppModuleBasic: AppModuleBasic{},
|
||||||
|
metrics: types.NewMetrics(telemetryOpts),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Name module name
|
||||||
|
func (am AppModule) Name() string {
|
||||||
|
return am.AppModuleBasic.Name()
|
||||||
|
}
|
||||||
|
|
||||||
|
// RegisterInvariants register module invariants
|
||||||
|
func (am AppModule) RegisterInvariants(_ sdk.InvariantRegistry) {}
|
||||||
|
|
||||||
|
// Route module message route name
|
||||||
|
// Deprecated: unused but necessary to fulfill AppModule interface
|
||||||
|
func (am AppModule) Route() sdk.Route { return sdk.Route{} }
|
||||||
|
|
||||||
|
// QuerierRoute module querier route name
|
||||||
|
// Deprecated: unused but necessary to fulfill AppModule interface
|
||||||
|
func (AppModule) QuerierRoute() string { return types.ModuleName }
|
||||||
|
|
||||||
|
// LegacyQuerierHandler returns no sdk.Querier.
|
||||||
|
// Deprecated: unused but necessary to fulfill AppModule interface
|
||||||
|
func (am AppModule) LegacyQuerierHandler(_ *codec.LegacyAmino) sdk.Querier {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// ConsensusVersion implements AppModule/ConsensusVersion.
|
||||||
|
func (AppModule) ConsensusVersion() uint64 { return 1 }
|
||||||
|
|
||||||
|
// RegisterServices registers module services.
|
||||||
|
func (am AppModule) RegisterServices(cfg module.Configurator) {}
|
||||||
|
|
||||||
|
// InitGenesis module init-genesis
|
||||||
|
func (am AppModule) InitGenesis(ctx sdk.Context, _ codec.JSONCodec, _ json.RawMessage) []abci.ValidatorUpdate {
|
||||||
|
return []abci.ValidatorUpdate{}
|
||||||
|
}
|
||||||
|
|
||||||
|
// ExportGenesis module export genesis
|
||||||
|
func (am AppModule) ExportGenesis(_ sdk.Context, cdc codec.JSONCodec) json.RawMessage {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
// BeginBlock module begin-block
|
||||||
|
func (am AppModule) BeginBlock(ctx sdk.Context, req abci.RequestBeginBlock) {
|
||||||
|
BeginBlocker(ctx, am.metrics)
|
||||||
|
}
|
||||||
|
|
||||||
|
// EndBlock module end-block
|
||||||
|
func (am AppModule) EndBlock(_ sdk.Context, _ abci.RequestEndBlock) []abci.ValidatorUpdate {
|
||||||
|
return []abci.ValidatorUpdate{}
|
||||||
|
}
|
36
x/metrics/spec/README.md
Normal file
36
x/metrics/spec/README.md
Normal file
@ -0,0 +1,36 @@
|
|||||||
|
<!--
|
||||||
|
order: 0
|
||||||
|
title: "Metrics Overview"
|
||||||
|
parent:
|
||||||
|
title: "metrics"
|
||||||
|
-->
|
||||||
|
|
||||||
|
# `metrics`
|
||||||
|
|
||||||
|
|
||||||
|
## Abstract
|
||||||
|
|
||||||
|
`x/metrics` is a stateless module that does not affect consensus. It captures chain metrics and emits them when the `instrumentation.prometheus` option is enabled in `config.toml`.
|
||||||
|
|
||||||
|
## Precision
|
||||||
|
|
||||||
|
The metrics emitted by `x/metrics` are `float64`s. They use `github.com/go-kit/kit/metrics` Prometheus gauges. Cosmos-sdk's `telemetry` package was not used because, at the time of writing, it only supports `float32`s and so does not maintain accurate representations of ints larger than ~16.8M. With `float64`s, integers may be accurately represented up to ~9e15.
|
||||||
|
|
||||||
|
## Metrics
|
||||||
|
|
||||||
|
The following metrics are defined:
|
||||||
|
* `cometbft_blocksync_latest_block_height` - this emulates the blocksync `latest_block_height` metric in CometBFT v0.38+. The `cometbft` namespace comes from the `instrumentation.namespace` config.toml value.
|
||||||
|
|
||||||
|
## Metric Labels
|
||||||
|
|
||||||
|
All metrics emitted have the labels defined in app.toml's `telemetry.global-labels` field. This is the same field used by cosmos-sdk's `telemetry` package.
|
||||||
|
|
||||||
|
example:
|
||||||
|
```toml
|
||||||
|
# app.toml
|
||||||
|
[telemetry]
|
||||||
|
global-labels = [
|
||||||
|
["chain_id", "kava_2222-10"],
|
||||||
|
["my_label", "my_value"],
|
||||||
|
]
|
||||||
|
```
|
6
x/metrics/types/keys.go
Normal file
6
x/metrics/types/keys.go
Normal file
@ -0,0 +1,6 @@
|
|||||||
|
package types
|
||||||
|
|
||||||
|
const (
|
||||||
|
// Name of the module
|
||||||
|
ModuleName = "metrics"
|
||||||
|
)
|
89
x/metrics/types/metrics.go
Normal file
89
x/metrics/types/metrics.go
Normal file
@ -0,0 +1,89 @@
|
|||||||
|
package types
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/go-kit/kit/metrics"
|
||||||
|
"github.com/go-kit/kit/metrics/discard"
|
||||||
|
prometheus "github.com/go-kit/kit/metrics/prometheus"
|
||||||
|
stdprometheus "github.com/prometheus/client_golang/prometheus"
|
||||||
|
"github.com/spf13/cast"
|
||||||
|
|
||||||
|
servertypes "github.com/cosmos/cosmos-sdk/server/types"
|
||||||
|
)
|
||||||
|
|
||||||
|
// TelemetryOptions defines the app configurations for the x/metrics module
|
||||||
|
type TelemetryOptions struct {
|
||||||
|
// CometBFT config value for instrumentation.prometheus (config.toml)
|
||||||
|
PrometheusEnabled bool
|
||||||
|
// Namespace for CometBFT metrics. Used to emulate CometBFT metrics.
|
||||||
|
CometBFTMetricsNamespace string
|
||||||
|
// A list of keys and values used as labels on all metrics
|
||||||
|
GlobalLabelsAndValues []string
|
||||||
|
}
|
||||||
|
|
||||||
|
// TelemetryOptionsFromAppOpts creates the TelemetryOptions from server AppOptions
|
||||||
|
func TelemetryOptionsFromAppOpts(appOpts servertypes.AppOptions) TelemetryOptions {
|
||||||
|
prometheusEnabled := cast.ToBool(appOpts.Get("instrumentation.prometheus"))
|
||||||
|
if !prometheusEnabled {
|
||||||
|
return TelemetryOptions{
|
||||||
|
GlobalLabelsAndValues: []string{},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// parse the app.toml global-labels into a slice of alternating label & value strings
|
||||||
|
// the value is expected to be a list of [label, value] tuples.
|
||||||
|
rawLabels := cast.ToSlice(appOpts.Get("telemetry.global-labels"))
|
||||||
|
globalLabelsAndValues := make([]string, 0, len(rawLabels)*2)
|
||||||
|
for _, gl := range rawLabels {
|
||||||
|
l := cast.ToStringSlice(gl)
|
||||||
|
globalLabelsAndValues = append(globalLabelsAndValues, l[0], l[1])
|
||||||
|
}
|
||||||
|
|
||||||
|
return TelemetryOptions{
|
||||||
|
PrometheusEnabled: true,
|
||||||
|
CometBFTMetricsNamespace: cast.ToString(appOpts.Get("instrumentation.namespace")),
|
||||||
|
GlobalLabelsAndValues: globalLabelsAndValues,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Metrics contains metrics exposed by this module.
|
||||||
|
// They use go-kit metrics like CometBFT as opposed to using cosmos-sdk telemetry
|
||||||
|
// because the sdk's telemetry only supports float32s, whereas go-kit prometheus
|
||||||
|
// metrics correctly handle float64s (and thus a larger number of int64s)
|
||||||
|
type Metrics struct {
|
||||||
|
// The height of the latest block.
|
||||||
|
// This gauges exactly emulates the default blocksync metric in CometBFT v0.38+
|
||||||
|
// It should be removed when kava has been updated to CometBFT v0.38+.
|
||||||
|
// see https://github.com/cometbft/cometbft/blob/v0.38.0-rc3/blocksync/metrics.gen.go
|
||||||
|
LatestBlockHeight metrics.Gauge
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewMetrics creates a new Metrics object based on whether or not prometheus instrumentation is enabled.
|
||||||
|
func NewMetrics(opts TelemetryOptions) *Metrics {
|
||||||
|
if opts.PrometheusEnabled {
|
||||||
|
return PrometheusMetrics(opts)
|
||||||
|
}
|
||||||
|
return NoopMetrics()
|
||||||
|
}
|
||||||
|
|
||||||
|
// PrometheusMetrics returns the gauges for when prometheus instrumentation is enabled.
|
||||||
|
func PrometheusMetrics(opts TelemetryOptions) *Metrics {
|
||||||
|
labels := []string{}
|
||||||
|
for i := 0; i < len(opts.GlobalLabelsAndValues); i += 2 {
|
||||||
|
labels = append(labels, opts.GlobalLabelsAndValues[i])
|
||||||
|
}
|
||||||
|
return &Metrics{
|
||||||
|
LatestBlockHeight: prometheus.NewGaugeFrom(stdprometheus.GaugeOpts{
|
||||||
|
Namespace: opts.CometBFTMetricsNamespace,
|
||||||
|
Subsystem: "blocksync",
|
||||||
|
Name: "latest_block_height",
|
||||||
|
Help: "The height of the latest block.",
|
||||||
|
}, labels).With(opts.GlobalLabelsAndValues...),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NoopMetrics are a do-nothing placeholder used when prometheus instrumentation is not enabled.
|
||||||
|
func NoopMetrics() *Metrics {
|
||||||
|
return &Metrics{
|
||||||
|
LatestBlockHeight: discard.NewGauge(),
|
||||||
|
}
|
||||||
|
}
|
72
x/metrics/types/metrics_test.go
Normal file
72
x/metrics/types/metrics_test.go
Normal file
@ -0,0 +1,72 @@
|
|||||||
|
package types_test
|
||||||
|
|
||||||
|
import (
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/go-kit/kit/metrics"
|
||||||
|
"github.com/go-kit/kit/metrics/prometheus"
|
||||||
|
"github.com/kava-labs/kava/x/metrics/types"
|
||||||
|
"github.com/stretchr/testify/require"
|
||||||
|
)
|
||||||
|
|
||||||
|
func isPrometheusGauge(g metrics.Gauge) bool {
|
||||||
|
_, ok := g.(*prometheus.Gauge)
|
||||||
|
return ok
|
||||||
|
}
|
||||||
|
|
||||||
|
var (
|
||||||
|
disabledOpts = types.TelemetryOptions{
|
||||||
|
PrometheusEnabled: false,
|
||||||
|
}
|
||||||
|
enabledOpts = types.TelemetryOptions{
|
||||||
|
PrometheusEnabled: true,
|
||||||
|
CometBFTMetricsNamespace: "cometbft",
|
||||||
|
GlobalLabelsAndValues: []string{"label1", "value1", "label2", "value2"},
|
||||||
|
}
|
||||||
|
)
|
||||||
|
|
||||||
|
func TestNewMetrics_DisabledVsEnabled(t *testing.T) {
|
||||||
|
myMetrics := types.NewMetrics(disabledOpts)
|
||||||
|
require.False(t, isPrometheusGauge(myMetrics.LatestBlockHeight))
|
||||||
|
|
||||||
|
myMetrics = types.NewMetrics(enabledOpts)
|
||||||
|
require.True(t, isPrometheusGauge(myMetrics.LatestBlockHeight))
|
||||||
|
}
|
||||||
|
|
||||||
|
type MockAppOpts struct {
|
||||||
|
store map[string]interface{}
|
||||||
|
}
|
||||||
|
|
||||||
|
func (mao *MockAppOpts) Get(key string) interface{} {
|
||||||
|
return mao.store[key]
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestTelemetryOptionsFromAppOpts(t *testing.T) {
|
||||||
|
appOpts := MockAppOpts{store: make(map[string]interface{})}
|
||||||
|
|
||||||
|
// test disabled functionality
|
||||||
|
appOpts.store["instrumentation.prometheus"] = false
|
||||||
|
|
||||||
|
opts := types.TelemetryOptionsFromAppOpts(&appOpts)
|
||||||
|
require.False(t, opts.PrometheusEnabled)
|
||||||
|
|
||||||
|
// test enabled functionality
|
||||||
|
appOpts.store["instrumentation.prometheus"] = true
|
||||||
|
appOpts.store["instrumentation.namespace"] = "magic"
|
||||||
|
appOpts.store["telemetry.global-labels"] = []interface{}{}
|
||||||
|
|
||||||
|
opts = types.TelemetryOptionsFromAppOpts(&appOpts)
|
||||||
|
require.True(t, opts.PrometheusEnabled)
|
||||||
|
require.Equal(t, "magic", opts.CometBFTMetricsNamespace)
|
||||||
|
require.Len(t, opts.GlobalLabelsAndValues, 0)
|
||||||
|
|
||||||
|
appOpts.store["telemetry.global-labels"] = []interface{}{
|
||||||
|
[]interface{}{"label1", "value1"},
|
||||||
|
[]interface{}{"label2", "value2"},
|
||||||
|
}
|
||||||
|
opts = types.TelemetryOptionsFromAppOpts(&appOpts)
|
||||||
|
require.True(t, opts.PrometheusEnabled)
|
||||||
|
require.Equal(t, "magic", opts.CometBFTMetricsNamespace)
|
||||||
|
require.Len(t, opts.GlobalLabelsAndValues, 4)
|
||||||
|
require.Equal(t, enabledOpts.GlobalLabelsAndValues, opts.GlobalLabelsAndValues)
|
||||||
|
}
|
Loading…
Reference in New Issue
Block a user