177 lines
4.0 KiB
Go
177 lines
4.0 KiB
Go
//go:build !freebsd
|
|
|
|
package testutil
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"strings"
|
|
|
|
dockercontainer "github.com/docker/docker/api/types/container"
|
|
"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
|
|
HostConfigModifier func(*dockercontainer.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,
|
|
HostConfigModifier: c.HostConfigModifier,
|
|
Cmd: c.Cmd,
|
|
Image: c.Image,
|
|
Name: c.Name,
|
|
Hostname: c.Hostname,
|
|
Networks: c.Networks,
|
|
WaitingFor: c.WaitingFor,
|
|
},
|
|
Started: true,
|
|
}
|
|
|
|
container, err := testcontainers.GenericContainer(c.ctx, req)
|
|
if err != nil {
|
|
return fmt.Errorf("container failed to start: %w", err)
|
|
}
|
|
c.container = container
|
|
|
|
c.Logs = TestLogConsumer{
|
|
Msgs: []string{},
|
|
}
|
|
c.container.FollowOutput(&c.Logs)
|
|
err = c.container.StartLogProducer(c.ctx)
|
|
if err != nil {
|
|
return fmt.Errorf("log producer failed: %w", err)
|
|
}
|
|
|
|
c.Address = "localhost"
|
|
|
|
err = c.LookupMappedPorts()
|
|
if 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() {
|
|
err := c.container.StopLogProducer()
|
|
if err != nil {
|
|
fmt.Println(err)
|
|
}
|
|
|
|
err = c.container.Terminate(c.ctx)
|
|
if 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())
|
|
}
|