cli: ensure opened sinks are properly closed

zap never closes open sinks except its own tests. This behaviour
prevents TestHandleLoggingParams from successful cleanup because
temp log output file can't be closed due to the following error:
```
TempDir RemoveAll cleanup: remove C:\\Users\\Anna\\AppData\\Local\\Temp\\TestHandleLoggingParams_debug5796883
33\\001\\file.log: The process cannot access the file because it is being used by another process.
```

So this tremendous cludge is made mosetly for our testing code.
It is not for concurrent usage (we don't have cases of
multithreaded access to logger output sink).
This commit is contained in:
Anna Shaleva 2022-02-07 18:51:54 +03:00 committed by Anna Shaleva
parent 59fdb69b9a
commit 2736d62d47
2 changed files with 51 additions and 15 deletions

View file

@ -31,9 +31,12 @@ import (
"go.uber.org/zap/zapcore" "go.uber.org/zap/zapcore"
) )
// _winfileSinkRegistered denotes whether zap has registered var (
// user-supplied factory for all sinks with `winfile`-prefixed scheme. // _winfileSinkRegistered denotes whether zap has registered
var _winfileSinkRegistered bool // user-supplied factory for all sinks with `winfile`-prefixed scheme.
_winfileSinkRegistered bool
_winfileSinkCloser func() error
)
// NewCommands returns 'node' command. // NewCommands returns 'node' command.
func NewCommands() []cli.Command { func NewCommands() []cli.Command {
@ -130,7 +133,9 @@ func getConfigFromContext(ctx *cli.Context) (config.Config, error) {
// handleLoggingParams reads logging parameters. // handleLoggingParams reads logging parameters.
// If user selected debug level -- function enables it. // If user selected debug level -- function enables it.
// If logPath is configured -- function creates dir and file for logging. // If logPath is configured -- function creates dir and file for logging.
func handleLoggingParams(ctx *cli.Context, cfg config.ApplicationConfiguration) (*zap.Logger, error) { // If logPath is configured on Windows -- function returns closer to be
// able to close sink for opened log output file.
func handleLoggingParams(ctx *cli.Context, cfg config.ApplicationConfiguration) (*zap.Logger, func() error, error) {
level := zapcore.InfoLevel level := zapcore.InfoLevel
if ctx.Bool("debug") { if ctx.Bool("debug") {
level = zapcore.DebugLevel level = zapcore.DebugLevel
@ -148,7 +153,7 @@ func handleLoggingParams(ctx *cli.Context, cfg config.ApplicationConfiguration)
if logPath := cfg.LogPath; logPath != "" { if logPath := cfg.LogPath; logPath != "" {
if err := io.MakeDirForFile(logPath, "logger"); err != nil { if err := io.MakeDirForFile(logPath, "logger"); err != nil {
return nil, err return nil, nil, err
} }
if runtime.GOOS == "windows" { if runtime.GOOS == "windows" {
@ -177,11 +182,16 @@ func handleLoggingParams(ctx *cli.Context, cfg config.ApplicationConfiguration)
case "stderr": case "stderr":
return os.Stderr, nil return os.Stderr, nil
} }
return os.OpenFile(u.Path[1:], // Remove leading slash left after url.Parse. f, err := os.OpenFile(u.Path[1:], // Remove leading slash left after url.Parse.
os.O_WRONLY|os.O_APPEND|os.O_CREATE, 0644) os.O_WRONLY|os.O_APPEND|os.O_CREATE, 0644)
_winfileSinkCloser = func() error {
_winfileSinkCloser = nil
return f.Close()
}
return f, err
}) })
if err != nil { if err != nil {
return nil, fmt.Errorf("failed to register windows-specific sinc: %w", err) return nil, nil, fmt.Errorf("failed to register windows-specific sinc: %w", err)
} }
_winfileSinkRegistered = true _winfileSinkRegistered = true
} }
@ -191,7 +201,8 @@ func handleLoggingParams(ctx *cli.Context, cfg config.ApplicationConfiguration)
cc.OutputPaths = []string{logPath} cc.OutputPaths = []string{logPath}
} }
return cc.Build() log, err := cc.Build()
return log, _winfileSinkCloser, err
} }
func initBCWithMetrics(cfg config.Config, log *zap.Logger) (*core.Blockchain, *metrics.Service, *metrics.Service, error) { func initBCWithMetrics(cfg config.Config, log *zap.Logger) (*core.Blockchain, *metrics.Service, *metrics.Service, error) {
@ -215,10 +226,13 @@ func dumpDB(ctx *cli.Context) error {
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
log, err := handleLoggingParams(ctx, cfg.ApplicationConfiguration) log, logCloser, err := handleLoggingParams(ctx, cfg.ApplicationConfiguration)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
if logCloser != nil {
defer func() { _ = logCloser() }()
}
count := uint32(ctx.Uint("count")) count := uint32(ctx.Uint("count"))
start := uint32(ctx.Uint("start")) start := uint32(ctx.Uint("start"))
@ -262,10 +276,13 @@ func restoreDB(ctx *cli.Context) error {
if err != nil { if err != nil {
return err return err
} }
log, err := handleLoggingParams(ctx, cfg.ApplicationConfiguration) log, logCloser, err := handleLoggingParams(ctx, cfg.ApplicationConfiguration)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
if logCloser != nil {
defer func() { _ = logCloser() }()
}
count := uint32(ctx.Uint("count")) count := uint32(ctx.Uint("count"))
var inStream = os.Stdin var inStream = os.Stdin
@ -438,10 +455,13 @@ func startServer(ctx *cli.Context) error {
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
log, err := handleLoggingParams(ctx, cfg.ApplicationConfiguration) log, logCloser, err := handleLoggingParams(ctx, cfg.ApplicationConfiguration)
if err != nil { if err != nil {
return cli.NewExitError(err, 1) return cli.NewExitError(err, 1)
} }
if logCloser != nil {
defer func() { _ = logCloser() }()
}
grace, cancel := context.WithCancel(newGraceContext()) grace, cancel := context.WithCancel(newGraceContext())
defer cancel() defer cancel()

View file

@ -52,8 +52,9 @@ func TestHandleLoggingParams(t *testing.T) {
cfg := config.ApplicationConfiguration{ cfg := config.ApplicationConfiguration{
LogPath: filepath.Join(logfile, "file.log"), LogPath: filepath.Join(logfile, "file.log"),
} }
_, err := handleLoggingParams(ctx, cfg) _, closer, err := handleLoggingParams(ctx, cfg)
require.Error(t, err) require.Error(t, err)
require.Nil(t, closer)
}) })
t.Run("default", func(t *testing.T) { t.Run("default", func(t *testing.T) {
@ -62,8 +63,13 @@ func TestHandleLoggingParams(t *testing.T) {
cfg := config.ApplicationConfiguration{ cfg := config.ApplicationConfiguration{
LogPath: testLog, LogPath: testLog,
} }
logger, err := handleLoggingParams(ctx, cfg) logger, closer, err := handleLoggingParams(ctx, cfg)
require.NoError(t, err) require.NoError(t, err)
t.Cleanup(func() {
if closer != nil {
require.NoError(t, closer())
}
})
require.True(t, logger.Core().Enabled(zap.InfoLevel)) require.True(t, logger.Core().Enabled(zap.InfoLevel))
require.False(t, logger.Core().Enabled(zap.DebugLevel)) require.False(t, logger.Core().Enabled(zap.DebugLevel))
}) })
@ -75,8 +81,13 @@ func TestHandleLoggingParams(t *testing.T) {
cfg := config.ApplicationConfiguration{ cfg := config.ApplicationConfiguration{
LogPath: testLog, LogPath: testLog,
} }
logger, err := handleLoggingParams(ctx, cfg) logger, closer, err := handleLoggingParams(ctx, cfg)
require.NoError(t, err) require.NoError(t, err)
t.Cleanup(func() {
if closer != nil {
require.NoError(t, closer())
}
})
require.True(t, logger.Core().Enabled(zap.InfoLevel)) require.True(t, logger.Core().Enabled(zap.InfoLevel))
require.True(t, logger.Core().Enabled(zap.DebugLevel)) require.True(t, logger.Core().Enabled(zap.DebugLevel))
}) })
@ -95,8 +106,13 @@ func TestInitBCWithMetrics(t *testing.T) {
ctx := cli.NewContext(cli.NewApp(), set, nil) ctx := cli.NewContext(cli.NewApp(), set, nil)
cfg, err := getConfigFromContext(ctx) cfg, err := getConfigFromContext(ctx)
require.NoError(t, err) require.NoError(t, err)
logger, err := handleLoggingParams(ctx, cfg.ApplicationConfiguration) logger, closer, err := handleLoggingParams(ctx, cfg.ApplicationConfiguration)
require.NoError(t, err) require.NoError(t, err)
t.Cleanup(func() {
if closer != nil {
require.NoError(t, closer())
}
})
t.Run("bad store", func(t *testing.T) { t.Run("bad store", func(t *testing.T) {
_, _, _, err = initBCWithMetrics(config.Config{}, logger) _, _, _, err = initBCWithMetrics(config.Config{}, logger)