diff --git a/beacond/cmd/defaults.go b/beacond/cmd/defaults.go index 79e542e036..5d4abd0578 100644 --- a/beacond/cmd/defaults.go +++ b/beacond/cmd/defaults.go @@ -104,7 +104,7 @@ func DefaultComponents() []any { *BeaconBlockHeader, *BeaconState, *BeaconStateMarshallable, *ExecutionPayload, *ExecutionPayloadHeader, *KVStore, *Logger, ], - components.ProvideReportingService[*Logger], + components.ProvideReportingService[*ExecutionPayload, *PayloadAttributes, *Logger], components.ProvideCometBFTService[*Logger], components.ProvideServiceRegistry[ *AvailabilityStore, diff --git a/beacond/cmd/types.go b/beacond/cmd/types.go index 449bd573dd..a8fb789fda 100644 --- a/beacond/cmd/types.go +++ b/beacond/cmd/types.go @@ -162,7 +162,7 @@ type ( NodeAPIServer = server.Server[NodeAPIContext] // ReportingService is a type alias for the reporting service. - ReportingService = version.ReportingService + ReportingService = version.ReportingService[*ExecutionPayload, *PayloadAttributes] // SidecarFactory is a type alias for the sidecar factory. SidecarFactory = dablob.SidecarFactory[ diff --git a/mod/execution/pkg/client/ethclient/engine.go b/mod/execution/pkg/client/ethclient/engine.go index 3db74e302f..139367e3d8 100644 --- a/mod/execution/pkg/client/ethclient/engine.go +++ b/mod/execution/pkg/client/ethclient/engine.go @@ -27,6 +27,7 @@ import ( "github.com/berachain/beacon-kit/mod/primitives/pkg/common" "github.com/berachain/beacon-kit/mod/primitives/pkg/eip4844" "github.com/berachain/beacon-kit/mod/primitives/pkg/version" + "github.com/ethereum/go-ethereum/beacon/engine" ) /* -------------------------------------------------------------------------- */ @@ -180,8 +181,12 @@ func (s *Client[ExecutionPayloadT]) GetClientVersionV1( ctx context.Context, ) ([]engineprimitives.ClientVersionV1, error) { result := make([]engineprimitives.ClientVersionV1, 0) + + // NOTE: although the ethereum spec does not require us passing a + // clientversion as param, it seems some clients require it and even enfore + // a valid Code. if err := s.Call( - ctx, &result, GetClientVersionV1, nil, + ctx, &result, GetClientVersionV1, engine.ClientVersionV1{Code: "GE"}, ); err != nil { return nil, err } diff --git a/mod/node-core/pkg/components/reporting_service.go b/mod/node-core/pkg/components/reporting_service.go index 94df432f31..3fb034d574 100644 --- a/mod/node-core/pkg/components/reporting_service.go +++ b/mod/node-core/pkg/components/reporting_service.go @@ -22,28 +22,41 @@ package components import ( "cosmossdk.io/depinject" + "github.com/berachain/beacon-kit/mod/execution/pkg/client" "github.com/berachain/beacon-kit/mod/log" "github.com/berachain/beacon-kit/mod/node-core/pkg/components/metrics" "github.com/berachain/beacon-kit/mod/node-core/pkg/services/version" + "github.com/berachain/beacon-kit/mod/primitives/pkg/constraints" sdkversion "github.com/cosmos/cosmos-sdk/version" ) type ReportingServiceInput[ + ExecutionPayloadT constraints.EngineType[ExecutionPayloadT], + PayloadAttributesT client.PayloadAttributes, LoggerT log.AdvancedLogger[LoggerT], ] struct { depinject.In Logger LoggerT TelemetrySink *metrics.TelemetrySink + EngineClient *client.EngineClient[ + ExecutionPayloadT, + PayloadAttributesT, + ] } func ProvideReportingService[ + ExecutionPayloadT constraints.EngineType[ExecutionPayloadT], + PayloadAttributesT client.PayloadAttributes, LoggerT log.AdvancedLogger[LoggerT], ]( - in ReportingServiceInput[LoggerT], -) *ReportingService { - return version.NewReportingService( + in ReportingServiceInput[ExecutionPayloadT, PayloadAttributesT, LoggerT], +) *version.ReportingService[ExecutionPayloadT, PayloadAttributesT] { + return version.NewReportingService[ + ExecutionPayloadT, PayloadAttributesT, + ]( in.Logger.With("service", "reporting"), in.TelemetrySink, sdkversion.Version, + in.EngineClient, ) } diff --git a/mod/node-core/pkg/components/service_registry.go b/mod/node-core/pkg/components/service_registry.go index 332df87ad6..1d942070b2 100644 --- a/mod/node-core/pkg/components/service_registry.go +++ b/mod/node-core/pkg/components/service_registry.go @@ -35,6 +35,7 @@ import ( "github.com/berachain/beacon-kit/mod/node-api/server" "github.com/berachain/beacon-kit/mod/node-core/pkg/components/metrics" service "github.com/berachain/beacon-kit/mod/node-core/pkg/services/registry" + "github.com/berachain/beacon-kit/mod/node-core/pkg/services/version" "github.com/berachain/beacon-kit/mod/observability/pkg/telemetry" ) @@ -101,7 +102,7 @@ type ServiceRegistryInput[ ] Logger LoggerT NodeAPIServer *server.Server[NodeAPIContextT] - ReportingService *ReportingService + ReportingService *version.ReportingService[ExecutionPayloadT, *engineprimitives.PayloadAttributes[WithdrawalT]] TelemetrySink *metrics.TelemetrySink TelemetryService *telemetry.Service ValidatorService *validator.Service[ diff --git a/mod/node-core/pkg/components/types.go b/mod/node-core/pkg/components/types.go index 2abea7ccac..cf3e95053e 100644 --- a/mod/node-core/pkg/components/types.go +++ b/mod/node-core/pkg/components/types.go @@ -27,7 +27,6 @@ import ( consruntimetypes "github.com/berachain/beacon-kit/mod/consensus/pkg/types" engineprimitives "github.com/berachain/beacon-kit/mod/engine-primitives/pkg/engine-primitives" "github.com/berachain/beacon-kit/mod/node-core/pkg/components/signer" - "github.com/berachain/beacon-kit/mod/node-core/pkg/services/version" "github.com/berachain/beacon-kit/mod/primitives/pkg/async" "github.com/berachain/beacon-kit/mod/primitives/pkg/transition" "github.com/berachain/beacon-kit/mod/storage/pkg/manager" @@ -40,9 +39,6 @@ import ( type ( // DBManager is a type alias for the database manager. DBManager = manager.DBManager - - // ReportingService is a type alias for the reporting service. - ReportingService = version.ReportingService ) /* -------------------------------------------------------------------------- */ diff --git a/mod/node-core/pkg/services/version/metrics.go b/mod/node-core/pkg/services/version/metrics.go deleted file mode 100644 index c0d64d4539..0000000000 --- a/mod/node-core/pkg/services/version/metrics.go +++ /dev/null @@ -1,73 +0,0 @@ -// SPDX-License-Identifier: BUSL-1.1 -// -// Copyright (C) 2024, Berachain Foundation. All rights reserved. -// Use of this software is governed by the Business Source License included -// in the LICENSE file of this repository and at www.mariadb.com/bsl11. -// -// ANY USE OF THE LICENSED WORK IN VIOLATION OF THIS LICENSE WILL AUTOMATICALLY -// TERMINATE YOUR RIGHTS UNDER THIS LICENSE FOR THE CURRENT AND ALL OTHER -// VERSIONS OF THE LICENSED WORK. -// -// THIS LICENSE DOES NOT GRANT YOU ANY RIGHT IN ANY TRADEMARK OR LOGO OF -// LICENSOR OR ITS AFFILIATES (PROVIDED THAT YOU MAY USE A TRADEMARK OR LOGO OF -// LICENSOR AS EXPRESSLY REQUIRED BY THIS LICENSE). -// -// TO THE EXTENT PERMITTED BY APPLICABLE LAW, THE LICENSED WORK IS PROVIDED ON -// AN “AS IS” BASIS. LICENSOR HEREBY DISCLAIMS ALL WARRANTIES AND CONDITIONS, -// EXPRESS OR IMPLIED, INCLUDING (WITHOUT LIMITATION) WARRANTIES OF -// MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE, NON-INFRINGEMENT, AND -// TITLE. - -package version - -import ( - "fmt" - "runtime" - - "github.com/berachain/beacon-kit/mod/log" -) - -// versionMetrics holds metrics related to the version reporting. -type versionMetrics struct { - // system is the current system the node is running on. - system string - // logger is the logger used to log information about the version. - logger log.Logger - // sink is the telemetry sink used to report metrics. - sink TelemetrySink -} - -// newVersionMetrics creates a new instance of versionMetrics. -func newVersionMetrics( - logger log.Logger, - sink TelemetrySink, -) *versionMetrics { - return &versionMetrics{ - system: runtime.GOOS + "/" + runtime.GOARCH, - logger: logger, - sink: sink, - } -} - -// reportVersion increments the versionReported counter. -func (vm *versionMetrics) reportVersion(version string) { - vm.logger.Info(fmt.Sprintf(` - - - +==========================================================================+ - + ⭐️ Star BeaconKit on GitHub @ https://github.com/berachain/beacon-kit + - + 🧩 Your node is running version: %-40s+ - + 💾 Your system: %-57s+ - + 🦺 Please report issues @ https://github.com/berachain/beacon-kit/issues + - +==========================================================================+ - - -`, - version, - vm.system, - )) - vm.sink.IncrementCounter( - "beacon_kit.runtime.version.reported", - "version", version, "system", vm.system, - ) -} diff --git a/mod/node-core/pkg/services/version/types.go b/mod/node-core/pkg/services/version/types.go index 16fb08b0f4..9c2365c0b0 100644 --- a/mod/node-core/pkg/services/version/types.go +++ b/mod/node-core/pkg/services/version/types.go @@ -25,4 +25,7 @@ type TelemetrySink interface { // IncrementCounter increments a counter metric identified by the provided // keys. IncrementCounter(key string, args ...string) + // SetGauge sets a gauge metric to the specified value, identified by the + // provided keys. + SetGauge(key string, value int64, args ...string) } diff --git a/mod/node-core/pkg/services/version/version.go b/mod/node-core/pkg/services/version/version.go index b204bcc9e9..c0be468b99 100644 --- a/mod/node-core/pkg/services/version/version.go +++ b/mod/node-core/pkg/services/version/version.go @@ -22,9 +22,13 @@ package version import ( "context" + "fmt" + "runtime" "time" + "github.com/berachain/beacon-kit/mod/execution/pkg/client" "github.com/berachain/beacon-kit/mod/log" + "github.com/berachain/beacon-kit/mod/primitives/pkg/constraints" ) // defaultReportingInterval is the default interval at which the version is @@ -33,46 +37,60 @@ const defaultReportingInterval = 5 * time.Minute // ReportingService is a service that periodically logs the running chain // version. -type ReportingService struct { +type ReportingService[ + ExecutionPayloadT constraints.EngineType[ExecutionPayloadT], + PayloadAttributesT client.PayloadAttributes, +] struct { // logger is used to log information about the running chain version. logger log.Logger // version represents the current version of the running chain. version string // reportingInterval is the interval at which the version is reported. reportingInterval time.Duration - // metrics contains the metrics for the version service. - metrics *versionMetrics + // sink is the telemetry sink used to report metrics. + sink TelemetrySink + // client to query the execution layer + client *client.EngineClient[ExecutionPayloadT, PayloadAttributesT] } // NewReportingService creates a new VersionReporterService. -func NewReportingService( +func NewReportingService[ + ExecutionPayloadT constraints.EngineType[ExecutionPayloadT], + PayloadAttributesT client.PayloadAttributes, +]( logger log.Logger, telemetrySink TelemetrySink, version string, -) *ReportingService { - return &ReportingService{ + engineClient *client.EngineClient[ExecutionPayloadT, PayloadAttributesT], +) *ReportingService[ + ExecutionPayloadT, PayloadAttributesT, +] { + return &ReportingService[ + ExecutionPayloadT, PayloadAttributesT, + ]{ logger: logger, version: version, reportingInterval: defaultReportingInterval, - metrics: newVersionMetrics(logger, telemetrySink), + sink: telemetrySink, + client: engineClient, } } // Name returns the name of the service. -func (*ReportingService) Name() string { +func (*ReportingService[_, _]) Name() string { return "reporting" } // Start begins the periodic logging of the chain version. -func (v *ReportingService) Start(ctx context.Context) error { - ticker := time.NewTicker(v.reportingInterval) - v.metrics.reportVersion(v.version) +func (rs *ReportingService[_, _]) Start(ctx context.Context) error { + ticker := time.NewTicker(rs.reportingInterval) + rs.handleReport(ctx) go func() { defer ticker.Stop() for { select { case <-ticker.C: - v.metrics.reportVersion(v.version) + rs.handleReport(ctx) continue case <-ctx.Done(): return @@ -81,3 +99,52 @@ func (v *ReportingService) Start(ctx context.Context) error { }() return nil } + +func (rs *ReportingService[_, _]) handleReport(ctx context.Context) { + systemInfo := runtime.GOOS + "/" + runtime.GOARCH + + rs.logger.Info(fmt.Sprintf(` + + + +==========================================================================+ + + ⭐️ Star BeaconKit on GitHub @ https://github.com/berachain/beacon-kit + + + 🧩 Your node is running version: %-40s+ + + 💾 Your system: %-57s+ + + 🦺 Please report issues @ https://github.com/berachain/beacon-kit/issues + + +==========================================================================+ + + +`, + rs.version, + systemInfo, + )) + + // TODO: Delete this counter as it should be included in the new beacon_kit.runtime.version metric. + rs.sink.IncrementCounter( + "beacon_kit.runtime.version.reported", + "version", rs.version, "system", systemInfo, + ) + + // Get the client version from the execution layer. + info, err := rs.client.GetClientVersionV1(ctx) + if err != nil { + rs.logger.Error("Failed to get client version", "err", err) + return + } + rs.logger.Info("GetClientVersionV1", "info", info) + + // the spec says we should have at least one client version + if len(info) == 0 { + rs.logger.Warn("No client version returned") + return + } + + // Report the version to the telemetry sink and include labels for beacon node version and eth name and version + var args = [8]string{ + "version", rs.version, + "system", systemInfo, + "eth_version", info[0].Version, + "eth_name", info[0].Name, + } + rs.sink.SetGauge("beacon_kit.runtime.version", 1, args[:]...) +}