From f45e37ab16b56e5f09cbbcaf41154edb75b9782d Mon Sep 17 00:00:00 2001 From: Simon Zhu Date: Mon, 27 Sep 2021 15:22:03 -0700 Subject: [PATCH 1/5] add logging for sync --- engine/common/synchronization/engine.go | 10 +++++----- engine/consensus/compliance/core.go | 4 ++-- 2 files changed, 7 insertions(+), 7 deletions(-) diff --git a/engine/common/synchronization/engine.go b/engine/common/synchronization/engine.go index d8adc100963..36eeaaff7e1 100644 --- a/engine/common/synchronization/engine.go +++ b/engine/common/synchronization/engine.go @@ -286,14 +286,14 @@ func (e *Engine) processAvailableResponses() { // onSyncResponse processes a synchronization response. func (e *Engine) onSyncResponse(originID flow.Identifier, res *messages.SyncResponse) { - e.log.Debug().Str("origin_id", originID.String()).Msg("received sync response") + e.log.Info().Str("origin_id", originID.String()).Msg("received sync response") final := e.finalizedHeader.Get() e.core.HandleHeight(final, res.Height) } // onBlockResponse processes a response containing a specifically requested block. func (e *Engine) onBlockResponse(originID flow.Identifier, res *messages.BlockResponse) { - e.log.Debug().Str("origin_id", originID.String()).Msg("received block response") + e.log.Info().Str("origin_id", originID.String()).Msg("received block response") // process the blocks one by one for _, block := range res.Blocks { if !e.core.HandleBlock(block.Header) { @@ -353,7 +353,7 @@ func (e *Engine) pollHeight() { Nonce: rand.Uint64(), Height: head.Height, } - e.log.Debug(). + e.log.Info(). Uint64("height", req.Height). Uint64("range_nonce", req.Nonce). Msg("sending sync request") @@ -380,7 +380,7 @@ func (e *Engine) sendRequests(participants flow.IdentifierList, ranges []flow.Ra errs = multierror.Append(errs, fmt.Errorf("could not submit range request: %w", err)) continue } - e.log.Debug(). + e.log.Info(). Uint64("range_from", req.FromHeight). Uint64("range_to", req.ToHeight). Uint64("range_nonce", req.Nonce). @@ -399,7 +399,7 @@ func (e *Engine) sendRequests(participants flow.IdentifierList, ranges []flow.Ra errs = multierror.Append(errs, fmt.Errorf("could not submit batch request: %w", err)) continue } - e.log.Debug(). + e.log.Info(). Strs("block_ids", flow.IdentifierList(batch.BlockIDs).Strings()). Uint64("range_nonce", req.Nonce). Msg("batch requested") diff --git a/engine/consensus/compliance/core.go b/engine/consensus/compliance/core.go index 5d8a5ee7971..0bcfed81cf0 100644 --- a/engine/consensus/compliance/core.go +++ b/engine/consensus/compliance/core.go @@ -162,7 +162,7 @@ func (c *Core) OnBlockProposal(originID flow.Identifier, proposal *messages.Bloc ancestorHeight = ancestor.Header.Height - 1 } - log.Debug(). + log.Info(). Uint64("ancestor_height", ancestorHeight). Hex("ancestor_id", ancestorID[:]). Msg("requesting missing ancestor for proposal") @@ -182,7 +182,7 @@ func (c *Core) OnBlockProposal(originID flow.Identifier, proposal *messages.Bloc c.mempool.MempoolEntries(metrics.ResourceProposal, c.pending.Size()) - log.Debug().Msg("requesting missing parent for proposal") + log.Info().Msg("requesting missing parent for proposal") c.sync.RequestBlock(header.ParentID) From 50e20f442525d08df65b454cf0f1d0df006c6d2e Mon Sep 17 00:00:00 2001 From: Simon Zhu Date: Wed, 6 Oct 2021 03:24:30 -0700 Subject: [PATCH 2/5] Implement log level admin command --- admin/command_runner.go | 14 ++- .../staked_access_node_builder.go | 2 + cmd/node_builder.go | 16 +-- cmd/scaffold.go | 101 +++++++++++------- engine/common/synchronization/engine.go | 8 +- engine/consensus/compliance/core.go | 4 +- 6 files changed, 89 insertions(+), 56 deletions(-) diff --git a/admin/command_runner.go b/admin/command_runner.go index e73e605b78c..de1ea788440 100644 --- a/admin/command_runner.go +++ b/admin/command_runner.go @@ -22,10 +22,15 @@ import ( const CommandRunnerShutdownTimeout = 5 * time.Second -type CommandHandler func(ctx context.Context, data map[string]interface{}) error -type CommandValidator func(data map[string]interface{}) error +type CommandHandler func(ctx context.Context, request *CommandRequest) error +type CommandValidator func(request *CommandRequest) error type CommandRunnerOption func(*CommandRunner) +type CommandRequest struct { + Data map[string]interface{} + ValidatorData interface{} +} + func WithTLS(config *tls.Config) CommandRunnerOption { return func(r *CommandRunner) { r.tlsConfig = config @@ -242,14 +247,15 @@ func (r *CommandRunner) runAdminServer(ctx context.Context) error { func (r *CommandRunner) runCommand(ctx context.Context, command string, data map[string]interface{}) error { r.logger.Info().Str("command", command).Msg("received new command") + req := &CommandRequest{Data: data} if validator := r.getValidator(command); validator != nil { - if validationErr := validator(data); validationErr != nil { + if validationErr := validator(req); validationErr != nil { return status.Error(codes.InvalidArgument, validationErr.Error()) } } if handler := r.getHandler(command); handler != nil { - if handleErr := handler(ctx, data); handleErr != nil { + if handleErr := handler(ctx, req); handleErr != nil { if errors.Is(handleErr, context.Canceled) { return status.Error(codes.Canceled, "client canceled") } else if errors.Is(handleErr, context.DeadlineExceeded) { diff --git a/cmd/access/node_builder/staked_access_node_builder.go b/cmd/access/node_builder/staked_access_node_builder.go index 1d0ea2f55af..ec1bf45e743 100644 --- a/cmd/access/node_builder/staked_access_node_builder.go +++ b/cmd/access/node_builder/staked_access_node_builder.go @@ -90,6 +90,8 @@ func (builder *StakedAccessNodeBuilder) Initialize() error { return err } + builder.EnqueueAdminServerInit(ctx) + builder.EnqueueTracer() return nil diff --git a/cmd/node_builder.go b/cmd/node_builder.go index 41d795712da..b830180ada9 100644 --- a/cmd/node_builder.go +++ b/cmd/node_builder.go @@ -104,10 +104,10 @@ type NodeBuilder interface { // while for a node running as a library, the config fields are expected to be initialized by the caller. type BaseConfig struct { nodeIDHex string - adminAddr string - adminCert string - adminKey string - adminClientCAs string + AdminAddr string + AdminCert string + AdminKey string + AdminClientCAs string BindAddr string NodeRole string datadir string @@ -176,10 +176,10 @@ func DefaultBaseConfig() *BaseConfig { return &BaseConfig{ nodeIDHex: NotSet, - adminAddr: NotSet, - adminCert: NotSet, - adminKey: NotSet, - adminClientCAs: NotSet, + AdminAddr: NotSet, + AdminCert: NotSet, + AdminKey: NotSet, + AdminClientCAs: NotSet, BindAddr: NotSet, BootstrapDir: "bootstrap", datadir: datadir, diff --git a/cmd/scaffold.go b/cmd/scaffold.go index 8a89e3795f9..635969f6253 100644 --- a/cmd/scaffold.go +++ b/cmd/scaffold.go @@ -138,10 +138,10 @@ func (fnb *FlowNodeBuilder) BaseFlags() { fnb.flags.UintVar(&fnb.BaseConfig.tracerSensitivity, "tracer-sensitivity", defaultConfig.tracerSensitivity, "adjusts the level of sampling when tracing is enabled. 0 means capture everything, higher value results in less samples") - fnb.flags.StringVar(&fnb.BaseConfig.adminAddr, "admin-addr", defaultConfig.adminAddr, "address to bind on for admin HTTP server") - fnb.flags.StringVar(&fnb.BaseConfig.adminCert, "admin-cert", defaultConfig.adminCert, "admin cert file (for TLS)") - fnb.flags.StringVar(&fnb.BaseConfig.adminKey, "admin-key", defaultConfig.adminKey, "admin key file (for TLS)") - fnb.flags.StringVar(&fnb.BaseConfig.adminClientCAs, "admin-client-certs", defaultConfig.adminClientCAs, "admin client certs (for mutual TLS)") + fnb.flags.StringVar(&fnb.BaseConfig.AdminAddr, "admin-addr", defaultConfig.AdminAddr, "address to bind on for admin HTTP server") + fnb.flags.StringVar(&fnb.BaseConfig.AdminCert, "admin-cert", defaultConfig.AdminCert, "admin cert file (for TLS)") + fnb.flags.StringVar(&fnb.BaseConfig.AdminKey, "admin-key", defaultConfig.AdminKey, "admin key file (for TLS)") + fnb.flags.StringVar(&fnb.BaseConfig.AdminClientCAs, "admin-client-certs", defaultConfig.AdminClientCAs, "admin client certs (for mutual TLS)") fnb.flags.DurationVar(&fnb.BaseConfig.DNSCacheTTL, "dns-cache-ttl", dns.DefaultTimeToLive, "time-to-live for dns cache") fnb.flags.UintVar(&fnb.BaseConfig.guaranteesCacheSize, "guarantees-cache-size", bstorage.DefaultCacheSize, "collection guarantees cache size") @@ -261,37 +261,44 @@ func (fnb *FlowNodeBuilder) EnqueueMetricsServerInit() { } func (fnb *FlowNodeBuilder) EnqueueAdminServerInit(ctx context.Context) { - fnb.Component("admin server", func(builder NodeBuilder, node *NodeConfig) (module.ReadyDoneAware, error) { - var opts []admin.CommandRunnerOption + if fnb.AdminAddr != NotSet { + if (fnb.AdminCert != NotSet || fnb.AdminKey != NotSet || fnb.AdminClientCAs != NotSet) && + !(fnb.AdminCert != NotSet && fnb.AdminKey != NotSet && fnb.AdminClientCAs != NotSet) { + fnb.Logger.Fatal().Msg("admin cert / key and client certs must all be provided to enable mutual TLS") + } + fnb.RegisterDefaultAdminCommands() + fnb.Component("admin server", func(builder NodeBuilder, node *NodeConfig) (module.ReadyDoneAware, error) { + var opts []admin.CommandRunnerOption - if node.adminCert != NotSet { - serverCert, err := tls.LoadX509KeyPair(node.adminCert, node.adminKey) - if err != nil { - return nil, err + if node.AdminCert != NotSet { + serverCert, err := tls.LoadX509KeyPair(node.AdminCert, node.AdminKey) + if err != nil { + return nil, err + } + clientCAs, err := ioutil.ReadFile(node.AdminClientCAs) + if err != nil { + return nil, err + } + certPool := x509.NewCertPool() + certPool.AppendCertsFromPEM(clientCAs) + config := &tls.Config{ + MinVersion: tls.VersionTLS13, + Certificates: []tls.Certificate{serverCert}, + ClientAuth: tls.RequireAndVerifyClientCert, + ClientCAs: certPool, + } + + opts = append(opts, admin.WithTLS(config)) } - clientCAs, err := ioutil.ReadFile(node.adminClientCAs) - if err != nil { + + command_runner := fnb.adminCommandBootstrapper.Bootstrap(fnb.Logger, fnb.AdminAddr, opts...) + if err := command_runner.Start(ctx); err != nil { return nil, err } - certPool := x509.NewCertPool() - certPool.AppendCertsFromPEM(clientCAs) - config := &tls.Config{ - MinVersion: tls.VersionTLS13, - Certificates: []tls.Certificate{serverCert}, - ClientAuth: tls.RequireAndVerifyClientCert, - ClientCAs: certPool, - } - - opts = append(opts, admin.WithTLS(config)) - } - - command_runner := fnb.adminCommandBootstrapper.Bootstrap(fnb.Logger, fnb.adminAddr, opts...) - if err := command_runner.Start(ctx); err != nil { - return nil, err - } - return command_runner, nil - }) + return command_runner, nil + }) + } } func (fnb *FlowNodeBuilder) RegisterBadgerMetrics() error { @@ -363,7 +370,8 @@ func (fnb *FlowNodeBuilder) initLogger() { if err != nil { log.Fatal().Err(err).Msg("invalid log level") } - log = log.Level(lvl) + log = log.Level(zerolog.DebugLevel) + zerolog.SetGlobalLevel(lvl) fnb.Logger = log } @@ -915,19 +923,36 @@ func (fnb *FlowNodeBuilder) Initialize() error { } } - if fnb.adminAddr != NotSet { - if (fnb.adminCert != NotSet || fnb.adminKey != NotSet || fnb.adminClientCAs != NotSet) && - !(fnb.adminCert != NotSet && fnb.adminKey != NotSet && fnb.adminClientCAs != NotSet) { - fnb.Logger.Fatal().Msg("admin cert / key and client certs must all be provided to enable mutual TLS") - } - fnb.EnqueueAdminServerInit(ctx) - } + fnb.EnqueueAdminServerInit(ctx) fnb.EnqueueTracer() return nil } +func (fnb *FlowNodeBuilder) RegisterDefaultAdminCommands() { + fnb.AdminCommand("set-log-level", func(ctx context.Context, req *admin.CommandRequest) error { + level := req.ValidatorData.(zerolog.Level) + zerolog.SetGlobalLevel(level) + return nil + }, func(req *admin.CommandRequest) error { + level, ok := req.Data["level"] + if !ok { + return errors.New("the \"level\" field must be provided") + } + levelStr, ok := level.(string) + if !ok { + return errors.New("\"level\" must be a string") + } + logLevel, err := zerolog.ParseLevel(levelStr) + if err != nil { + return fmt.Errorf("failed to parse level: %w", err) + } + req.ValidatorData = logLevel + return nil + }) +} + // Run calls Ready() to start all the node modules and components. It also sets up a channel to gracefully shut // down each component if a SIGINT is received. Until a SIGINT is received, Run will block. // Since, Run is a blocking call it should only be used when running a node as it's own independent process. diff --git a/engine/common/synchronization/engine.go b/engine/common/synchronization/engine.go index 36eeaaff7e1..744dd97d914 100644 --- a/engine/common/synchronization/engine.go +++ b/engine/common/synchronization/engine.go @@ -286,14 +286,14 @@ func (e *Engine) processAvailableResponses() { // onSyncResponse processes a synchronization response. func (e *Engine) onSyncResponse(originID flow.Identifier, res *messages.SyncResponse) { - e.log.Info().Str("origin_id", originID.String()).Msg("received sync response") + e.log.Debug().Str("origin_id", originID.String()).Msg("received sync response") final := e.finalizedHeader.Get() e.core.HandleHeight(final, res.Height) } // onBlockResponse processes a response containing a specifically requested block. func (e *Engine) onBlockResponse(originID flow.Identifier, res *messages.BlockResponse) { - e.log.Info().Str("origin_id", originID.String()).Msg("received block response") + e.log.Debug().Str("origin_id", originID.String()).Msg("received block response") // process the blocks one by one for _, block := range res.Blocks { if !e.core.HandleBlock(block.Header) { @@ -353,7 +353,7 @@ func (e *Engine) pollHeight() { Nonce: rand.Uint64(), Height: head.Height, } - e.log.Info(). + e.log.Debug(). Uint64("height", req.Height). Uint64("range_nonce", req.Nonce). Msg("sending sync request") @@ -399,7 +399,7 @@ func (e *Engine) sendRequests(participants flow.IdentifierList, ranges []flow.Ra errs = multierror.Append(errs, fmt.Errorf("could not submit batch request: %w", err)) continue } - e.log.Info(). + e.log.Debug(). Strs("block_ids", flow.IdentifierList(batch.BlockIDs).Strings()). Uint64("range_nonce", req.Nonce). Msg("batch requested") diff --git a/engine/consensus/compliance/core.go b/engine/consensus/compliance/core.go index 0bcfed81cf0..5d8a5ee7971 100644 --- a/engine/consensus/compliance/core.go +++ b/engine/consensus/compliance/core.go @@ -162,7 +162,7 @@ func (c *Core) OnBlockProposal(originID flow.Identifier, proposal *messages.Bloc ancestorHeight = ancestor.Header.Height - 1 } - log.Info(). + log.Debug(). Uint64("ancestor_height", ancestorHeight). Hex("ancestor_id", ancestorID[:]). Msg("requesting missing ancestor for proposal") @@ -182,7 +182,7 @@ func (c *Core) OnBlockProposal(originID flow.Identifier, proposal *messages.Bloc c.mempool.MempoolEntries(metrics.ResourceProposal, c.pending.Size()) - log.Info().Msg("requesting missing parent for proposal") + log.Debug().Msg("requesting missing parent for proposal") c.sync.RequestBlock(header.ParentID) From 3c9febd91b1bd0da54a9e6e5581b07da9f296830 Mon Sep 17 00:00:00 2001 From: Simon Zhu Date: Wed, 6 Oct 2021 03:31:07 -0700 Subject: [PATCH 3/5] Update command_runner_test.go --- admin/command_runner_test.go | 24 ++++++++++++------------ 1 file changed, 12 insertions(+), 12 deletions(-) diff --git a/admin/command_runner_test.go b/admin/command_runner_test.go index e2a59139dd6..a775bf51eb4 100644 --- a/admin/command_runner_test.go +++ b/admin/command_runner_test.go @@ -86,15 +86,15 @@ func (suite *CommandRunnerSuite) SetupCommandRunner(opts ...CommandRunnerOption) func (suite *CommandRunnerSuite) TestHandler() { called := false - suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, data map[string]interface{}) error { + suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, req *CommandRequest) error { select { case <-ctx.Done(): return ctx.Err() default: } - suite.EqualValues(data["string"], "foo") - suite.EqualValues(data["number"], 123) + suite.EqualValues(req.Data["string"], "foo") + suite.EqualValues(req.Data["number"], 123) called = true return nil @@ -142,7 +142,7 @@ func (suite *CommandRunnerSuite) TestUnimplementedHandler() { func (suite *CommandRunnerSuite) TestValidator() { calls := 0 - suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, data map[string]interface{}) error { + suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, req *CommandRequest) error { select { case <-ctx.Done(): return ctx.Err() @@ -155,8 +155,8 @@ func (suite *CommandRunnerSuite) TestValidator() { }) validatorErr := errors.New("unexpected value") - suite.bootstrapper.RegisterValidator("foo", func(data map[string]interface{}) error { - if data["key"] != "value" { + suite.bootstrapper.RegisterValidator("foo", func(req *CommandRequest) error { + if req.Data["key"] != "value" { return validatorErr } return nil @@ -192,7 +192,7 @@ func (suite *CommandRunnerSuite) TestValidator() { func (suite *CommandRunnerSuite) TestHandlerError() { handlerErr := errors.New("handler error") - suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, data map[string]interface{}) error { + suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, req *CommandRequest) error { select { case <-ctx.Done(): return ctx.Err() @@ -222,7 +222,7 @@ func (suite *CommandRunnerSuite) TestHandlerError() { } func (suite *CommandRunnerSuite) TestTimeout() { - suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, data map[string]interface{}) error { + suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, req *CommandRequest) error { <-ctx.Done() return ctx.Err() }) @@ -248,14 +248,14 @@ func (suite *CommandRunnerSuite) TestTimeout() { func (suite *CommandRunnerSuite) TestHTTPServer() { called := false - suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, data map[string]interface{}) error { + suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, req *CommandRequest) error { select { case <-ctx.Done(): return ctx.Err() default: } - suite.EqualValues(data["key"], "value") + suite.EqualValues(req.Data["key"], "value") called = true return nil @@ -387,14 +387,14 @@ func generateCerts(t *testing.T) (tls.Certificate, *x509.CertPool, tls.Certifica func (suite *CommandRunnerSuite) TestTLS() { called := false - suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, data map[string]interface{}) error { + suite.bootstrapper.RegisterHandler("foo", func(ctx context.Context, req *CommandRequest) error { select { case <-ctx.Done(): return ctx.Err() default: } - suite.EqualValues(data["key"], "value") + suite.EqualValues(req.Data["key"], "value") called = true return nil From 5cb80c9b6269c5d1cbf5d26d1b75fd9c6b4f0766 Mon Sep 17 00:00:00 2001 From: Simon Zhu Date: Wed, 6 Oct 2021 12:26:03 -0700 Subject: [PATCH 4/5] Move command into new file --- admin/commands/command.go | 8 ++++++ admin/commands/common/set_log_level.go | 35 ++++++++++++++++++++++++++ cmd/scaffold.go | 22 ++-------------- 3 files changed, 45 insertions(+), 20 deletions(-) create mode 100644 admin/commands/command.go create mode 100644 admin/commands/common/set_log_level.go diff --git a/admin/commands/command.go b/admin/commands/command.go new file mode 100644 index 00000000000..52ebaf00779 --- /dev/null +++ b/admin/commands/command.go @@ -0,0 +1,8 @@ +package commands + +import "github.com/onflow/flow-go/admin" + +type AdminCommand struct { + Handler admin.CommandHandler + Validator admin.CommandValidator +} diff --git a/admin/commands/common/set_log_level.go b/admin/commands/common/set_log_level.go new file mode 100644 index 00000000000..db06f61f15a --- /dev/null +++ b/admin/commands/common/set_log_level.go @@ -0,0 +1,35 @@ +package common + +import ( + "context" + "errors" + "fmt" + + "github.com/onflow/flow-go/admin" + "github.com/onflow/flow-go/admin/commands" + "github.com/rs/zerolog" +) + +var SetLogLevelCommand commands.AdminCommand = commands.AdminCommand{ + Handler: func(ctx context.Context, req *admin.CommandRequest) error { + level := req.ValidatorData.(zerolog.Level) + zerolog.SetGlobalLevel(level) + return nil + }, + Validator: func(req *admin.CommandRequest) error { + level, ok := req.Data["level"] + if !ok { + return errors.New("the \"level\" field must be provided") + } + levelStr, ok := level.(string) + if !ok { + return errors.New("\"level\" must be a string") + } + logLevel, err := zerolog.ParseLevel(levelStr) + if err != nil { + return fmt.Errorf("failed to parse level: %w", err) + } + req.ValidatorData = logLevel + return nil + }, +} diff --git a/cmd/scaffold.go b/cmd/scaffold.go index 635969f6253..3d14419c0c5 100644 --- a/cmd/scaffold.go +++ b/cmd/scaffold.go @@ -22,6 +22,7 @@ import ( "github.com/spf13/pflag" "github.com/onflow/flow-go/admin" + "github.com/onflow/flow-go/admin/commands/common" "github.com/onflow/flow-go/cmd/build" "github.com/onflow/flow-go/fvm" "github.com/onflow/flow-go/model/bootstrap" @@ -931,26 +932,7 @@ func (fnb *FlowNodeBuilder) Initialize() error { } func (fnb *FlowNodeBuilder) RegisterDefaultAdminCommands() { - fnb.AdminCommand("set-log-level", func(ctx context.Context, req *admin.CommandRequest) error { - level := req.ValidatorData.(zerolog.Level) - zerolog.SetGlobalLevel(level) - return nil - }, func(req *admin.CommandRequest) error { - level, ok := req.Data["level"] - if !ok { - return errors.New("the \"level\" field must be provided") - } - levelStr, ok := level.(string) - if !ok { - return errors.New("\"level\" must be a string") - } - logLevel, err := zerolog.ParseLevel(levelStr) - if err != nil { - return fmt.Errorf("failed to parse level: %w", err) - } - req.ValidatorData = logLevel - return nil - }) + fnb.AdminCommand("set-log-level", common.SetLogLevelCommand.Handler, common.SetLogLevelCommand.Validator) } // Run calls Ready() to start all the node modules and components. It also sets up a channel to gracefully shut From 041127c92f08138249b25477e3e5baa9688c3c39 Mon Sep 17 00:00:00 2001 From: Simon Zhu Date: Wed, 6 Oct 2021 19:36:47 -0700 Subject: [PATCH 5/5] fix lint --- admin/commands/common/set_log_level.go | 3 ++- 1 file changed, 2 insertions(+), 1 deletion(-) diff --git a/admin/commands/common/set_log_level.go b/admin/commands/common/set_log_level.go index db06f61f15a..04a2c748b60 100644 --- a/admin/commands/common/set_log_level.go +++ b/admin/commands/common/set_log_level.go @@ -5,9 +5,10 @@ import ( "errors" "fmt" + "github.com/rs/zerolog" + "github.com/onflow/flow-go/admin" "github.com/onflow/flow-go/admin/commands" - "github.com/rs/zerolog" ) var SetLogLevelCommand commands.AdminCommand = commands.AdminCommand{