Add structured logging infrastructure (#246)
New global flags:
* `--log-file FILE`: can be literal `stdout`, `stderr`, or a file name (default `stderr`)
* `--log-level LEVEL`: can be `error`, `warn`, `info`, `debug`, `trace`, or `disabled` (default `disabled`)
* `--log-format TYPE`: can be `text` or `json` (default `text`)
New functions in the `log` package take a `context.Context` and retrieve
the logger from said context.
Because we carry the logger in a context, adding
[attributes](https://pkg.go.dev/golang.org/x/exp/slog#hdr-Attrs_and_Values)
to the logger can be done as follows:
```go
ctx = log.NewContext(ctx, log.GetLogger(ctx).With("foo", "bar"))
```
2023-03-16 13:46:53 +00:00
|
|
|
package log
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"fmt"
|
2023-08-15 13:50:40 +00:00
|
|
|
"log/slog"
|
Add structured logging infrastructure (#246)
New global flags:
* `--log-file FILE`: can be literal `stdout`, `stderr`, or a file name (default `stderr`)
* `--log-level LEVEL`: can be `error`, `warn`, `info`, `debug`, `trace`, or `disabled` (default `disabled`)
* `--log-format TYPE`: can be `text` or `json` (default `text`)
New functions in the `log` package take a `context.Context` and retrieve
the logger from said context.
Because we carry the logger in a context, adding
[attributes](https://pkg.go.dev/golang.org/x/exp/slog#hdr-Attrs_and_Values)
to the logger can be done as follows:
```go
ctx = log.NewContext(ctx, log.GetLogger(ctx).With("foo", "bar"))
```
2023-03-16 13:46:53 +00:00
|
|
|
"runtime"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
sdk "github.com/databricks/databricks-sdk-go/logger"
|
|
|
|
)
|
|
|
|
|
|
|
|
// slogAdapter makes an slog.Logger usable with the Databricks SDK.
|
|
|
|
type slogAdapter struct{}
|
|
|
|
|
|
|
|
func (s slogAdapter) Enabled(ctx context.Context, level sdk.Level) bool {
|
|
|
|
logger := GetLogger(ctx)
|
|
|
|
switch level {
|
|
|
|
case sdk.LevelTrace:
|
|
|
|
return logger.Enabled(ctx, LevelTrace)
|
|
|
|
case sdk.LevelDebug:
|
|
|
|
return logger.Enabled(ctx, LevelDebug)
|
|
|
|
case sdk.LevelInfo:
|
|
|
|
return logger.Enabled(ctx, LevelInfo)
|
|
|
|
case sdk.LevelWarn:
|
|
|
|
return logger.Enabled(ctx, LevelWarn)
|
|
|
|
case sdk.LevelError:
|
|
|
|
return logger.Enabled(ctx, LevelError)
|
|
|
|
default:
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s slogAdapter) log(logger *slog.Logger, ctx context.Context, level slog.Level, msg string) {
|
|
|
|
var pcs [1]uintptr
|
|
|
|
// skip [runtime.Callers, this function, this function's caller, the caller in the SDK].
|
|
|
|
runtime.Callers(4, pcs[:])
|
|
|
|
r := slog.NewRecord(time.Now(), level, msg, pcs[0])
|
|
|
|
r.AddAttrs(slog.Bool("sdk", true))
|
|
|
|
if ctx == nil {
|
|
|
|
ctx = context.Background()
|
|
|
|
}
|
|
|
|
_ = logger.Handler().Handle(ctx, r)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s slogAdapter) Tracef(ctx context.Context, format string, v ...any) {
|
|
|
|
logger := GetLogger(ctx)
|
|
|
|
if !logger.Enabled(ctx, LevelTrace) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
s.log(logger, ctx, LevelTrace, fmt.Sprintf(format, v...))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s slogAdapter) Debugf(ctx context.Context, format string, v ...any) {
|
|
|
|
logger := GetLogger(ctx)
|
|
|
|
if !logger.Enabled(ctx, LevelDebug) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
s.log(logger, ctx, LevelDebug, fmt.Sprintf(format, v...))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s slogAdapter) Infof(ctx context.Context, format string, v ...any) {
|
|
|
|
logger := GetLogger(ctx)
|
|
|
|
if !logger.Enabled(ctx, LevelInfo) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
s.log(logger, ctx, LevelInfo, fmt.Sprintf(format, v...))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s slogAdapter) Warnf(ctx context.Context, format string, v ...any) {
|
|
|
|
logger := GetLogger(ctx)
|
|
|
|
if !logger.Enabled(ctx, LevelWarn) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
s.log(logger, ctx, LevelWarn, fmt.Sprintf(format, v...))
|
|
|
|
}
|
|
|
|
|
|
|
|
func (s slogAdapter) Errorf(ctx context.Context, format string, v ...any) {
|
|
|
|
logger := GetLogger(ctx)
|
|
|
|
if !logger.Enabled(ctx, LevelError) {
|
|
|
|
return
|
|
|
|
}
|
|
|
|
s.log(logger, ctx, LevelError, fmt.Sprintf(format, v...))
|
|
|
|
}
|
|
|
|
|
|
|
|
func init() {
|
|
|
|
// Configure SDK to use this logger.
|
|
|
|
sdk.DefaultLogger = slogAdapter{}
|
|
|
|
}
|