Skip to content

Commit

Permalink
feat: WithLogger ContainerCustomizer support (testcontainers#2259)
Browse files Browse the repository at this point in the history
* feat: WithLogger ContainerCustomizer support

Add support to use WithLogger as a ContainerCustomizer so that callers
have an easy way to configure the logger of a container.

Add tests for WithLogger.

Validate Logger and LoggerOption implement the required interfaces.

* chore: add note about order

Add a note about WithLogger being used before other functions so it can capture any logging they may generate.

Co-authored-by: Manuel de la Peña <social.mdelapenya@gmail.com>

---------

Co-authored-by: Manuel de la Peña <social.mdelapenya@gmail.com>
  • Loading branch information
stevenh and mdelapenya authored Mar 1, 2024
1 parent 8838003 commit c474bcd
Show file tree
Hide file tree
Showing 3 changed files with 80 additions and 4 deletions.
23 changes: 23 additions & 0 deletions docs/features/common_functional_options.md
Original file line number Diff line number Diff line change
Expand Up @@ -41,6 +41,29 @@ func (g *TestLogConsumer) Accept(l Log) {
}
```

#### WithLogger

- Not available until the next release of testcontainers-go <a href="https://github.com/testcontainers/testcontainers-go"><span class="tc-version">:material-tag: main</span></a>

If you need to either pass logger to a container, you can use `testcontainers.WithLogger`.

!!!info
Consider calling this before other "With" functions as these may generate logs.

In this example we also use `TestLogger` which writes to the passed in `testing.TB` using `Logf`.
The result is that we capture all logging from the container into the test context meaning its
hidden behind `go test -v` and is associated with the relevant test, providing the user with
useful context instead of appearing out of band.

```golang
func TestHandler(t *testing.T) {
logger := TestLogger(t)
_, err := postgresModule.RunContainer(ctx, testcontainers.WithLogger(logger))
require.NoError(t, err)
// Do something with container.
}
```

Please read the [Following Container Logs](/features/follow_logs) documentation for more information about creating log consumers.

#### Wait Strategies
Expand Down
30 changes: 26 additions & 4 deletions logger.go
Original file line number Diff line number Diff line change
Expand Up @@ -12,6 +12,14 @@ import (
// Logger is the default log instance
var Logger Logging = log.New(os.Stderr, "", log.LstdFlags)

// Validate our types implement the required interfaces.
var (
_ Logging = (*log.Logger)(nil)
_ ContainerCustomizer = LoggerOption{}
_ GenericProviderOption = LoggerOption{}
_ DockerProviderOption = LoggerOption{}
)

// Logging defines the Logger interface
type Logging interface {
Printf(format string, v ...interface{})
Expand All @@ -24,37 +32,51 @@ func LogDockerServerInfo(ctx context.Context, client client.APIClient, logger Lo
}

// TestLogger returns a Logging implementation for testing.TB
// This way logs from testcontainers are part of the test output of a test suite or test case
// This way logs from testcontainers are part of the test output of a test suite or test case.
func TestLogger(tb testing.TB) Logging {
tb.Helper()
return testLogger{TB: tb}
}

// WithLogger is a generic option that implements GenericProviderOption, DockerProviderOption
// It replaces the global Logging implementation with a user defined one e.g. to aggregate logs from testcontainers
// with the logs of specific test case
// WithLogger returns a generic option that sets the logger to be used.
//
// Consider calling this before other "With functions" as these may generate logs.
//
// This can be given a TestLogger to collect the logs from testcontainers into a
// test case.
func WithLogger(logger Logging) LoggerOption {
return LoggerOption{
logger: logger,
}
}

// LoggerOption is a generic option that sets the logger to be used.
//
// It can be used to set the logger for providers and containers.
type LoggerOption struct {
logger Logging
}

// ApplyGenericTo implements GenericProviderOption.
func (o LoggerOption) ApplyGenericTo(opts *GenericProviderOptions) {
opts.Logger = o.logger
}

// ApplyDockerTo implements DockerProviderOption.
func (o LoggerOption) ApplyDockerTo(opts *DockerProviderOptions) {
opts.Logger = o.logger
}

// Customize implements ContainerCustomizer.
func (o LoggerOption) Customize(req *GenericContainerRequest) {
req.Logger = o.logger
}

type testLogger struct {
testing.TB
}

// Printf implements Logging.
func (t testLogger) Printf(format string, v ...interface{}) {
t.Helper()
t.Logf(format, v...)
Expand Down
31 changes: 31 additions & 0 deletions logger_test.go
Original file line number Diff line number Diff line change
@@ -0,0 +1,31 @@
package testcontainers

import (
"testing"

"github.com/stretchr/testify/require"
)

func TestWithLogger(t *testing.T) {
logger := TestLogger(t)
logOpt := WithLogger(logger)
t.Run("container", func(t *testing.T) {
var req GenericContainerRequest
logOpt.Customize(&req)
require.Equal(t, logger, req.Logger)
})

t.Run("provider", func(t *testing.T) {
var opts GenericProviderOptions
logOpt.ApplyGenericTo(&opts)
require.Equal(t, logger, opts.Logger)
})

t.Run("docker", func(t *testing.T) {
opts := &DockerProviderOptions{
GenericProviderOptions: &GenericProviderOptions{},
}
logOpt.ApplyDockerTo(opts)
require.Equal(t, logger, opts.Logger)
})
}

0 comments on commit c474bcd

Please sign in to comment.