1
0
Fork 0
telegraf/testutil/container.go
Daniel Baumann 4978089aab
Adding upstream version 1.34.4.
Signed-off-by: Daniel Baumann <daniel@debian.org>
2025-05-24 07:26:29 +02:00

217 lines
5.1 KiB
Go

//go:build !freebsd
package testutil
import (
"context"
"fmt"
"io"
"strings"
"github.com/docker/docker/api/types/container"
"github.com/docker/docker/api/types/image"
"github.com/docker/go-connections/nat"
"github.com/testcontainers/testcontainers-go"
"github.com/testcontainers/testcontainers-go/wait"
)
type TestLogConsumer struct {
Msgs []string
}
func (g *TestLogConsumer) Accept(l testcontainers.Log) {
g.Msgs = append(g.Msgs, string(l.Content))
}
type Container struct {
Entrypoint []string
Env map[string]string
Files map[string]string
HostAccessPorts []int
HostConfigModifier func(*container.HostConfig)
ExposedPorts []string
Cmd []string
Image string
Name string
Hostname string
Networks []string
WaitingFor wait.Strategy
Address string
Ports map[string]string
Logs TestLogConsumer
container testcontainers.Container
ctx context.Context
}
func (c *Container) Start() error {
c.ctx = context.Background()
files := make([]testcontainers.ContainerFile, 0, len(c.Files))
for k, v := range c.Files {
files = append(files, testcontainers.ContainerFile{
ContainerFilePath: k,
HostFilePath: v,
FileMode: 0o755,
})
}
req := testcontainers.GenericContainerRequest{
ContainerRequest: testcontainers.ContainerRequest{
Entrypoint: c.Entrypoint,
Env: c.Env,
ExposedPorts: c.ExposedPorts,
Files: files,
HostAccessPorts: c.HostAccessPorts,
HostConfigModifier: c.HostConfigModifier,
Cmd: c.Cmd,
Image: c.Image,
Name: c.Name,
Hostname: c.Hostname,
Networks: c.Networks,
WaitingFor: c.WaitingFor,
},
Started: true,
}
cntnr, err := testcontainers.GenericContainer(c.ctx, req)
if err != nil {
return fmt.Errorf("container failed to start: %w", err)
}
c.container = cntnr
c.Logs = TestLogConsumer{}
c.container.FollowOutput(&c.Logs)
if err := c.container.StartLogProducer(c.ctx); err != nil {
return fmt.Errorf("log producer failed: %w", err)
}
c.Address = "localhost"
info, err := c.GetInfo()
if err != nil {
return fmt.Errorf("getting info failed: %w", err)
}
fmt.Println("Started container:", info)
if err := c.LookupMappedPorts(); err != nil {
c.Terminate()
return fmt.Errorf("port lookup failed: %w", err)
}
return nil
}
// LookupMappedPorts creates a lookup table of exposed ports to mapped ports
func (c *Container) LookupMappedPorts() error {
if len(c.ExposedPorts) == 0 {
return nil
}
if len(c.Ports) == 0 {
c.Ports = make(map[string]string)
}
for _, port := range c.ExposedPorts {
// strip off leading host port: 80:8080 -> 8080
if strings.Contains(port, ":") {
port = strings.Split(port, ":")[1]
}
p, err := c.container.MappedPort(c.ctx, nat.Port(port))
if err != nil {
return fmt.Errorf("failed to find %q: %w", port, err)
}
// strip off the transport: 80/tcp -> 80
if strings.Contains(port, "/") {
port = strings.Split(port, "/")[0]
}
fmt.Printf("mapped container port %q to host port %q\n", port, p.Port())
c.Ports[port] = p.Port()
}
return nil
}
func (c *Container) Exec(cmds []string) (int, io.Reader, error) {
return c.container.Exec(c.ctx, cmds)
}
func (c *Container) PrintLogs() {
fmt.Println("--- Container Logs Start ---")
for _, msg := range c.Logs.Msgs {
fmt.Print(msg)
}
fmt.Println("--- Container Logs End ---")
}
func (c *Container) Terminate() {
if err := c.container.StopLogProducer(); err != nil {
fmt.Println(err)
}
if err := c.container.Terminate(c.ctx); err != nil {
fmt.Printf("failed to terminate the container: %s", err)
}
c.PrintLogs()
}
func (c *Container) Pause() error {
provider, err := testcontainers.NewDockerProvider()
if err != nil {
return fmt.Errorf("getting provider failed: %w", err)
}
return provider.Client().ContainerPause(c.ctx, c.container.GetContainerID())
}
func (c *Container) Resume() error {
provider, err := testcontainers.NewDockerProvider()
if err != nil {
return fmt.Errorf("getting provider failed: %w", err)
}
return provider.Client().ContainerUnpause(c.ctx, c.container.GetContainerID())
}
func (c *Container) GetInfo() (string, error) {
dc, ok := c.container.(*testcontainers.DockerContainer)
if !ok {
return "not a docker container", nil
}
ci, err := dc.Inspect(c.ctx)
if err != nil {
return "", fmt.Errorf("inspecting container failed: %w", err)
}
provider, err := testcontainers.NewDockerProvider()
if err != nil {
return "", fmt.Errorf("getting provider failed: %w", err)
}
summaries, err := provider.Client().ImageList(c.ctx, image.ListOptions{})
if err != nil {
return "", fmt.Errorf("listing images failed: %w", err)
}
for _, s := range summaries {
if s.ID != ci.ContainerJSONBase.Image {
continue
}
var digest []string
for _, d := range s.RepoDigests {
if _, suffix, found := strings.Cut(d, "@"); found {
digest = append(digest, suffix)
} else {
digest = append(digest, d)
}
}
return fmt.Sprintf("%s (%s)", dc.Image, strings.Join(digest, ",")), nil
}
return "unknown", nil
}