Skip to content

Commit

Permalink
refactor: replace logger
Browse files Browse the repository at this point in the history
This replaces the zap logging engine with Go 1.21's structured
logger, log/slog, or more precisely a thin wrapper around that
("xlog").

The log/slog package has a few things missing, which are
present in xlog:

1. xlog provides a no-op logger, which simply discards any log output.
   This is extensively used in our tests.
2. xlog has a Fatal() output method, which simply calls Error() and then
   os.Exit(1).
3. xlog treats error values as first-class citizen. Since (log/slog).Error()
   is a convenience function for their default logger instance, there is
   no built-in way to treat errors as values. In comparison, (xlog).Error()
   constructs an slog.Attr, since xlog does not provide a default logger.

Point (2) is debatable, since xlog.Fatal is only used in cmd/texd/main.go,
so I'd be willing to forfeit it.

Some TODOs remain:

- xlog, i.e. its extension over log/slog, is not really tested
- documentation is missing
- the current xlog constructor (New) is a bit clunky to use,
  maybe switch to functional options?
- some tests create a local buffer as log output target - this
  could be made easier with a `log, buf := xlog.NewBuffered()`
  constructor (albeit, the overhead is minimal)
- for local development, I still like to have some colorization

Issue: #1
  • Loading branch information
dmke committed Oct 31, 2024
1 parent 7e9eb47 commit 62f9fda
Show file tree
Hide file tree
Showing 32 changed files with 369 additions and 246 deletions.
87 changes: 39 additions & 48 deletions cmd/texd/main.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,10 +4,12 @@ import (
"context"
"errors"
"fmt"
"log/slog"
"os"
"os/signal"
"runtime"
"runtime/debug"
"strings"
"sync"
"syscall"
"time"
Expand All @@ -20,11 +22,10 @@ import (
"github.com/digineo/texd/refstore/nop"
"github.com/digineo/texd/service"
"github.com/digineo/texd/tex"
"github.com/digineo/texd/xlog"
"github.com/docker/go-units"
"github.com/spf13/pflag"
"github.com/thediveo/enumflag"
"go.uber.org/zap"
"go.uber.org/zap/zapcore"
)

const (
Expand Down Expand Up @@ -53,7 +54,7 @@ var (
engine = tex.DefaultEngine.Name()
jobdir = ""
pull = false
logLevel = zapcore.InfoLevel.String()
logLevel = slog.LevelInfo.String()
maxJobSize = units.BytesSize(float64(opts.MaxJobSize))
storageDSN = ""
showVersion = false
Expand Down Expand Up @@ -119,7 +120,7 @@ func parseFlags(progname string, args ...string) []string {
fmt.Sprintf("enable reference store and configure with `DSN`, available adapters are: %v", refstore.AvailableAdapters()))
fs.BoolVar(&pull, "pull", pull, "always pull Docker images")
fs.StringVar(&logLevel, "log-level", logLevel,
"set logging verbosity, acceptable values are: [debug, info, warn, error, dpanic, panic, fatal]")
"set logging verbosity, acceptable values are: [debug, info, warn, error, fatal]")
fs.BoolVarP(&showVersion, "version", "v", showVersion,
`print version information and exit`)

Expand Down Expand Up @@ -148,8 +149,10 @@ func parseFlags(progname string, args ...string) []string {
func main() { //nolint:funlen
texd.PrintBanner(os.Stdout)
images := parseFlags(os.Args[0], os.Args[1:]...)
log, sync := setupLogger()
defer sync()
log, err := setupLogger()
if err != nil {
panic(err)
}

if showVersion {
printVersion()
Expand All @@ -158,32 +161,32 @@ func main() { //nolint:funlen

if err := tex.SetJobBaseDir(jobdir); err != nil {
log.Fatal("error setting job directory",
zap.String("flag", "--job-directory"),
zap.Error(err))
xlog.String("flag", "--job-directory"),
xlog.Error(err))
}
if err := tex.SetDefaultEngine(engine); err != nil {
log.Fatal("error setting default TeX engine",
zap.String("flag", "--tex-engine"),
zap.Error(err))
xlog.String("flag", "--tex-engine"),
xlog.Error(err))
}
if maxsz, err := units.FromHumanSize(maxJobSize); err != nil {
log.Fatal("error parsing maximum job size",
zap.String("flag", "--max-job-size"),
zap.Error(err))
xlog.String("flag", "--max-job-size"),
xlog.Error(err))
} else {
opts.MaxJobSize = maxsz
}
if storageDSN != "" {
rp, err := retentionPolicy()
if err != nil {
log.Fatal("error initializing retention policy",
zap.String("flag", "--retention-policy, and/or --rp-access-items, --rp-access-size"),
zap.Error(err))
xlog.String("flag", "--retention-policy, and/or --rp-access-items, --rp-access-size"),
xlog.Error(err))
}
if adapter, err := refstore.NewStore(storageDSN, rp); err != nil {
log.Fatal("error parsing reference store DSN",
zap.String("flag", "--reference-store"),
zap.Error(err))
xlog.String("flag", "--reference-store"),
xlog.Error(err))
} else {
opts.RefStore = adapter
}
Expand All @@ -192,37 +195,39 @@ func main() { //nolint:funlen
}

if len(images) > 0 {
log.Info("using docker", zap.Strings("images", images))
log.Info("using docker", xlog.String("images", strings.Join(images, ",")))
cli, err := exec.NewDockerClient(log, tex.JobBaseDir())
if err != nil {
log.Fatal("error connecting to dockerd", zap.Error(err))
log.Error("error connecting to dockerd", xlog.Error(err))
os.Exit(1)
}

opts.Images, err = cli.SetImages(context.Background(), pull, images...)
opts.Mode = "container"
if err != nil {
log.Fatal("error setting images", zap.Error(err))
log.Error("error setting images", xlog.Error(err))
os.Exit(1)
}
opts.Executor = cli.Executor
}

stop, err := service.Start(opts, log)
if err != nil {
log.Fatal("failed to start service", zap.Error(err))
log.Fatal("failed to start service", xlog.Error(err))
}
onExit(log, stop)
}

type stopFun func(context.Context) error

func onExit(log *zap.Logger, stopper ...stopFun) {
func onExit(log xlog.Logger, stopper ...stopFun) {
exitCh := make(chan os.Signal, 2) //nolint:mnd // idiomatic
signal.Notify(exitCh, syscall.SIGINT, syscall.SIGTERM)
sig := <-exitCh

log.Info("performing shutdown, press Ctrl+C to exit now",
zap.String("signal", sig.String()),
zap.Duration("graceful-wait-timeout", exitTimeout))
xlog.String("signal", sig.String()),
slog.Duration("graceful-wait-timeout", exitTimeout))

ctx, cancel := context.WithTimeout(context.Background(), exitTimeout)
defer cancel()
Expand All @@ -232,7 +237,7 @@ func onExit(log *zap.Logger, stopper ...stopFun) {
for _, stop := range stopper {
go func(f stopFun) {
if err := f(ctx); err != nil {
log.Error("error while shutting down", zap.Error(err))
log.Error("error while shutting down", xlog.Error(err))
}
wg.Done()
}(stop)
Expand Down Expand Up @@ -275,34 +280,20 @@ func printVersion() {
}
}

func setupLogger() (*zap.Logger, func()) {
var cfg zap.Config
if texd.Development() {
cfg = zap.NewDevelopmentConfig()
} else {
cfg = zap.NewProductionConfig()
}

lvl, lvlErr := zapcore.ParseLevel(logLevel)
if lvlErr == nil {
cfg.Level = zap.NewAtomicLevelAt(lvl)
}

log, err := cfg.Build()
func setupLogger() (xlog.Logger, error) {
lvl, err := xlog.ParseLevel(logLevel)
if err != nil {
// we don't have a logger yet, so logging the error
// proves to be complicated :)
panic(err)
return nil, err
}

if lvlErr != nil {
log.Error("error parsing log level",
zap.String("flag", "--log-level"),
zap.Error(lvlErr))
o := &slog.HandlerOptions{
AddSource: true,
// XXX: provide ReplaceAttr callback to normalize Source locations?
Level: lvl,
}

zap.ReplaceGlobals(log)
return log, func() {
_ = log.Sync()
if texd.Development() {
return xlog.New(xlog.TypeText, os.Stderr, o)
}
return xlog.New(xlog.TypeJSON, os.Stdout, o)
}
8 changes: 4 additions & 4 deletions exec/docker.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,7 @@ import (
"context"

"github.com/digineo/texd/tex"
"go.uber.org/zap"
"github.com/digineo/texd/xlog"
)

type dockerRunner interface {
Expand All @@ -23,17 +23,17 @@ func (dc *DockerClient) Executor(doc Document) Exec {
}
}

func (x *dockerExec) Run(ctx context.Context, log *zap.Logger) error {
func (x *dockerExec) Run(ctx context.Context, log xlog.Logger) error {
dir, cmd, err := x.extract()
if err != nil {
return tex.CompilationError("invalid document", err, nil)
}

tag := x.doc.Image()
log.Debug("running latexmk", zap.Strings("args", cmd[1:]))
log.Debug("running latexmk", xlog.Any("args", cmd[1:]))
output, err := x.cli.Run(ctx, tag, dir, cmd)
if err != nil {
log.Error("compilation failed", zap.Error(err))
log.Error("compilation failed", xlog.Error(err))
return tex.CompilationError("compilation failed", err, tex.KV{
"cmd": cmd[0],
"args": cmd[1:],
Expand Down
16 changes: 8 additions & 8 deletions exec/docker_client.go
Original file line number Diff line number Diff line change
Expand Up @@ -8,14 +8,14 @@ import (
"os"

"github.com/digineo/texd/service/middleware"
"github.com/digineo/texd/xlog"
"github.com/docker/docker/api/types/container"
"github.com/docker/docker/api/types/image"
"github.com/docker/docker/api/types/mount"
"github.com/docker/docker/client"
"github.com/docker/docker/pkg/jsonmessage"
"github.com/docker/docker/pkg/stdcopy"
"github.com/moby/term"
"go.uber.org/zap"
)

// newClient is swapped in tests.
Expand All @@ -26,7 +26,7 @@ var newClient = func() (client.APIClient, error) {
// DockerClient wraps a Docker client instance and provides methods to
// pull images and start containers.
type DockerClient struct {
log *zap.Logger
log xlog.Logger
cli client.APIClient
images []image.Summary

Expand All @@ -40,14 +40,14 @@ type DockerClient struct {
// When running in a Docker-in-Docker environment, baseDir is used to
// determine the volume path on the Docker host, in order to mount
// job directories correctly.
func NewDockerClient(log *zap.Logger, baseDir string) (h *DockerClient, err error) {
func NewDockerClient(log xlog.Logger, baseDir string) (h *DockerClient, err error) {
cli, err := newClient()
if err != nil {
return nil, err
}

if log == nil {
log = zap.NewNop()
log = xlog.NewNop()
}
dc := &DockerClient{
log: log,
Expand Down Expand Up @@ -82,14 +82,14 @@ func (dc *DockerClient) SetImages(ctx context.Context, alwaysPull bool, tags ...
if img.ID == "" || alwaysPull {
toPull = append(toPull, tag)
} else {
dc.log.Info("image already present", zap.String("image", tag))
dc.log.Info("image already present", xlog.String("image", tag))
knownImages = append(knownImages, img)
}
}

p := newProgressReporter(os.Stdout)
for _, tag := range toPull {
dc.log.Info("pulling missing image", zap.String("image", tag))
dc.log.Info("pulling missing image", xlog.String("image", tag))
if err := dc.pull(ctx, tag, p); err != nil {
return nil, err
}
Expand Down Expand Up @@ -209,8 +209,8 @@ func (dc *DockerClient) prepareContainer(ctx context.Context, tag, wd string, cm
}
dc.log.Debug("container is ready",
middleware.RequestIDField(ctx),
zap.String("id", worker.ID),
zap.String("work-dir", wd))
xlog.String("id", worker.ID),
xlog.String("work-dir", wd))
return worker.ID, nil
}

Expand Down
4 changes: 2 additions & 2 deletions exec/docker_client_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,7 @@ import (
"syscall"
"testing"

"github.com/digineo/texd/xlog"
"github.com/docker/docker/api/types"
"github.com/docker/docker/api/types/container"
"github.com/docker/docker/api/types/image"
Expand All @@ -22,7 +23,6 @@ import (
"github.com/spf13/afero"
"github.com/stretchr/testify/mock"
"github.com/stretchr/testify/suite"
"go.uber.org/zap"
)

// bg is used as default context given to the apiMock stubs.
Expand Down Expand Up @@ -115,7 +115,7 @@ func (s *dockerClientSuite) SetupTest() {
s.cli = &apiMock{}
s.subject = &DockerClient{
cli: s.cli,
log: zap.NewNop(),
log: xlog.NewNop(),
}
}

Expand Down
10 changes: 5 additions & 5 deletions exec/docker_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -7,10 +7,10 @@ import (
"testing"

"github.com/digineo/texd/tex"
"github.com/digineo/texd/xlog"
"github.com/stretchr/testify/assert"
"github.com/stretchr/testify/mock"
"github.com/stretchr/testify/require"
"go.uber.org/zap"
)

type dockerClientMock struct {
Expand All @@ -24,7 +24,7 @@ func (m *dockerClientMock) Run(ctx context.Context, tag, wd string, cmd []string

func TestDockerClient_Executor(t *testing.T) {
subject := (&DockerClient{
log: zap.NewNop(),
log: xlog.NewNop(),
cli: &apiMock{},
}).Executor(&mockDocument{})
require.NotNil(t, subject)
Expand All @@ -37,7 +37,7 @@ func TestDockerExec_invalidDocument(t *testing.T) {
cli: nil, // not accessed
}

err := exec.Run(bg, zap.NewNop())
err := exec.Run(bg, xlog.NewNop())
require.EqualError(t, err, "invalid document: "+io.ErrClosedPipe.Error())
}

Expand All @@ -55,7 +55,7 @@ func TestDockerExec_latexmkFailed(t *testing.T) {
cli: cli,
}

err := exec.Run(bg, zap.NewNop())
err := exec.Run(bg, xlog.NewNop())
require.EqualError(t, err, "compilation failed: "+errStart.Error())
assert.True(t, tex.IsCompilationError(err))

Expand All @@ -78,6 +78,6 @@ func TestDockerExec_success(t *testing.T) {
cli: cli,
}

err := exec.Run(bg, zap.NewNop())
err := exec.Run(bg, xlog.NewNop())
require.NoError(t, err)
}
4 changes: 2 additions & 2 deletions exec/exec.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,11 @@ import (
"context"

"github.com/digineo/texd/tex"
"go.uber.org/zap"
"github.com/digineo/texd/xlog"
)

type Exec interface {
Run(ctx context.Context, logger *zap.Logger) error
Run(ctx context.Context, logger xlog.Logger) error
}

// Document is a sub-set of the tex.Document interface.
Expand Down
Loading

0 comments on commit 62f9fda

Please sign in to comment.