Skip to content
New issue

Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.

By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.

Already on GitHub? Sign in to your account

feat: support reading Docker host from the current Docker context #2810

Open
wants to merge 56 commits into
base: main
Choose a base branch
from
Open
Show file tree
Hide file tree
Changes from all commits
Commits
Show all changes
56 commits
Select commit Hold shift + click to select a range
6c1b3d8
feat: support reading Docker context
mdelapenya Oct 2, 2024
415c896
chore: move read Docker config to the core
mdelapenya Oct 2, 2024
1e620f1
chore: read docker config more consistently
mdelapenya Oct 2, 2024
ea44c92
chore: extract to a parse function to detect remote docker hosts
mdelapenya Oct 2, 2024
549b1cb
docs: add Docker context to docs
mdelapenya Oct 2, 2024
fbefbdf
fix: lint
mdelapenya Oct 2, 2024
1ca3f63
chore: simplify container provider initialisation
mdelapenya Oct 3, 2024
7e1e4b5
feat: support configuring the bridge network name
mdelapenya Oct 3, 2024
2decbe3
chore: rename function
mdelapenya Oct 3, 2024
c596e37
fix: remove wrong copy&paste
mdelapenya Oct 3, 2024
ece6ea5
chore: simplify
mdelapenya Oct 3, 2024
a775c4b
chore: simplify error
mdelapenya Oct 3, 2024
5ac2263
chore: simple naming in test
mdelapenya Oct 3, 2024
932fce2
fix: do not lose the original error
mdelapenya Oct 3, 2024
9c048f7
fix: line endings
mdelapenya Oct 3, 2024
77cdda6
fix: case insensitive regex
mdelapenya Oct 4, 2024
7eab89f
chore: deprecate reaper_network and always use the bridge one
mdelapenya Oct 4, 2024
1e58328
chore: support replacing the bridge network in the endpoint modifier
mdelapenya Oct 4, 2024
8abfeba
fix: lint
mdelapenya Oct 4, 2024
203d762
chore: assume the container runtime uses the default network by default
mdelapenya Oct 4, 2024
464bc9b
chore: rename variable to avoid shading package
mdelapenya Oct 5, 2024
37a8142
fix: if the properties file does not exist, use default config
mdelapenya Oct 5, 2024
07003ff
fix: typo
mdelapenya Oct 5, 2024
adf0ba1
fix: use default config in tests
mdelapenya Oct 7, 2024
b31da78
docs: simplify
mdelapenya Oct 14, 2024
a38a345
chore: simplify error when docker context is not found
mdelapenya Oct 14, 2024
14aa3dd
fix: check if the docker socket is listening
mdelapenya Oct 16, 2024
b327978
docs: document the docker context support
mdelapenya Oct 16, 2024
33528c6
chore: remove bridge network custom configuration
mdelapenya Oct 16, 2024
ca07e04
Merge branch 'main' into context-support
mdelapenya Oct 18, 2024
dab5614
Merge branch 'main' into context-support
mdelapenya Oct 21, 2024
a313dc5
chore: use t.Helper
mdelapenya Oct 21, 2024
5511d17
Merge branch 'main' into context-support
mdelapenya Oct 28, 2024
b38e857
Merge branch 'main' into context-support
mdelapenya Nov 19, 2024
bc2f04b
chore: proper deprecation path
mdelapenya Nov 19, 2024
dc571d3
Merge branch 'main' into context-support
mdelapenya Nov 25, 2024
ad79b43
chore: comment out empty field used for documentation
mdelapenya Nov 25, 2024
4d4769f
chore: use require
mdelapenya Nov 25, 2024
56d6774
docs: update podman commands
mdelapenya Nov 25, 2024
1273e02
chore: readability in var comments
mdelapenya Nov 25, 2024
e7dd83c
chore: unique parsing error messages
mdelapenya Nov 25, 2024
584f6da
docs: readability as function comments
mdelapenya Nov 25, 2024
3b59086
chore: idiomatic var declaration
mdelapenya Nov 25, 2024
1c93fe8
chore: extract file name to constant
mdelapenya Nov 25, 2024
92d306f
chore: simplify default config
mdelapenya Nov 25, 2024
499a05d
chore: make errors private
mdelapenya Nov 25, 2024
cdb874d
chore: simplify function
mdelapenya Nov 25, 2024
1933f42
fix: remove unused func
mdelapenya Nov 25, 2024
f4783a7
chore: bubble up config errors when when config is incorrect
mdelapenya Dec 4, 2024
28f0802
chore: use better test names
mdelapenya Dec 4, 2024
f8d24c7
chore: deprecate ReadConfig in favour of NewConfig
mdelapenya Dec 5, 2024
786d79c
chore: line separator
mdelapenya Dec 5, 2024
4bd9934
chore: do not export var
mdelapenya Dec 5, 2024
5f7eb43
Merge branch 'main' into context-support
mdelapenya Dec 5, 2024
8fcd654
chore: bubble up error on environment configuration
mdelapenya Dec 5, 2024
13934da
chore: rename test cases
mdelapenya Dec 5, 2024
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
16 changes: 15 additions & 1 deletion config.go
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@ import (
"github.com/testcontainers/testcontainers-go/internal/config"
)

// Deprecated: use [testcontainers.Config] instead
// TestcontainersConfig represents the configuration for Testcontainers
type TestcontainersConfig struct {
Host string `properties:"docker.host,default="` // Deprecated: use Config.Host instead
Expand All @@ -14,10 +15,15 @@ type TestcontainersConfig struct {
Config config.Config
}

// Deprecated: use [testcontainers.NewConfig] instead
// ReadConfig reads from testcontainers properties file, storing the result in a singleton instance
// of the TestcontainersConfig struct
func ReadConfig() TestcontainersConfig {
cfg := config.Read()
cfg, err := NewConfig()
if err != nil {
return TestcontainersConfig{}
}

return TestcontainersConfig{
Host: cfg.Host,
TLSVerify: cfg.TLSVerify,
Expand All @@ -27,3 +33,11 @@ func ReadConfig() TestcontainersConfig {
Config: cfg,
}
}

// Config is a type alias for the internal config.Config type
type Config = config.Config

// NewConfig reads the properties file and returns a new Config instance
func NewConfig() (Config, error) {
return config.Read()
}
94 changes: 5 additions & 89 deletions docker.go
Original file line number Diff line number Diff line change
Expand Up @@ -43,9 +43,9 @@ import (
var _ Container = (*DockerContainer)(nil)

const (
Bridge = "bridge" // Bridge network name (as well as driver)
Podman = "podman"
ReaperDefault = "reaper_default" // Default network name when bridge is not available
Bridge = "bridge" // Deprecated, it will removed in the next major release. Bridge network driver and name
Podman = "podman" // Deprecated: Podman is supported through the current Docker context
ReaperDefault = "reaper_default" // Deprecated: it will removed in the next major release. Default network name when bridge is not available
packagePath = "github.com/testcontainers/testcontainers-go"
)

Expand Down Expand Up @@ -1026,29 +1026,6 @@ func (p *DockerProvider) CreateContainer(ctx context.Context, req ContainerReque
// defer the close of the Docker client connection the soonest
defer p.Close()

var defaultNetwork string
defaultNetwork, err = p.ensureDefaultNetwork(ctx)
if err != nil {
return nil, fmt.Errorf("ensure default network: %w", err)
}

// If default network is not bridge make sure it is attached to the request
// as container won't be attached to it automatically
// in case of Podman the bridge network is called 'podman' as 'bridge' would conflict
if defaultNetwork != p.defaultBridgeNetworkName {
isAttached := false
for _, net := range req.Networks {
if net == defaultNetwork {
isAttached = true
break
}
}

if !isAttached {
req.Networks = append(req.Networks, defaultNetwork)
}
}

imageName := req.Image

env := []string{}
Expand Down Expand Up @@ -1452,6 +1429,7 @@ func (p *DockerProvider) RunContainer(ctx context.Context, req ContainerRequest)
return c, nil
}

// Deprecated: use [testcontainers.NewConfig] instead
// Config provides the TestcontainersConfig read from $HOME/.testcontainers.properties or
// the environment variables
func (p *DockerProvider) Config() TestcontainersConfig {
Expand Down Expand Up @@ -1522,10 +1500,6 @@ func (p *DockerProvider) CreateNetwork(ctx context.Context, req NetworkRequest)
// defer the close of the Docker client connection the soonest
defer p.Close()

if _, err = p.ensureDefaultNetwork(ctx); err != nil {
return nil, fmt.Errorf("ensure default network: %w", err)
}

if req.Labels == nil {
req.Labels = make(map[string]string)
}
Expand Down Expand Up @@ -1593,13 +1567,7 @@ func (p *DockerProvider) GetNetwork(ctx context.Context, req NetworkRequest) (ne
}

func (p *DockerProvider) GetGatewayIP(ctx context.Context) (string, error) {
// Use a default network as defined in the DockerProvider
defaultNetwork, err := p.ensureDefaultNetwork(ctx)
if err != nil {
return "", fmt.Errorf("ensure default network: %w", err)
}

nw, err := p.GetNetwork(ctx, NetworkRequest{Name: defaultNetwork})
nw, err := p.GetNetwork(ctx, NetworkRequest{Name: Bridge})
stevenh marked this conversation as resolved.
Show resolved Hide resolved
if err != nil {
return "", err
}
Expand All @@ -1618,58 +1586,6 @@ func (p *DockerProvider) GetGatewayIP(ctx context.Context) (string, error) {
return ip, nil
}

// ensureDefaultNetwork ensures that defaultNetwork is set and creates
// it if it does not exist, returning its value.
// It is safe to call this method concurrently.
func (p *DockerProvider) ensureDefaultNetwork(ctx context.Context) (string, error) {
p.mtx.Lock()
defer p.mtx.Unlock()

if p.defaultNetwork != "" {
// Already set.
return p.defaultNetwork, nil
}

networkResources, err := p.client.NetworkList(ctx, network.ListOptions{})
if err != nil {
return "", fmt.Errorf("network list: %w", err)
}

// TODO: remove once we have docker context support via #2810
// Prefer the default bridge network if it exists.
// This makes the results stable as network list order is not guaranteed.
for _, net := range networkResources {
switch net.Name {
case p.defaultBridgeNetworkName:
p.defaultNetwork = p.defaultBridgeNetworkName
return p.defaultNetwork, nil
case ReaperDefault:
p.defaultNetwork = ReaperDefault
}
}

if p.defaultNetwork != "" {
return p.defaultNetwork, nil
}

// Create a bridge network for the container communications.
_, err = p.client.NetworkCreate(ctx, ReaperDefault, network.CreateOptions{
Driver: Bridge,
Attachable: true,
Labels: GenericLabels(),
})
// If the network already exists, we can ignore the error as that can
// happen if we are running multiple tests in parallel and we only
// need to ensure that the network exists.
if err != nil && !errdefs.IsConflict(err) {
return "", fmt.Errorf("network create: %w", err)
}

p.defaultNetwork = ReaperDefault

return p.defaultNetwork, nil
}

// ContainerFromType builds a Docker container struct from the response of the Docker API
func (p *DockerProvider) ContainerFromType(ctx context.Context, response types.Container) (ctr *DockerContainer, err error) {
exposedPorts := make([]string, len(response.Ports))
Expand Down
24 changes: 1 addition & 23 deletions docker_auth.go
Original file line number Diff line number Diff line change
Expand Up @@ -172,7 +172,7 @@ func configKey(cfg *dockercfg.Config) (string, error) {
// getDockerAuthConfigs returns a map with the auth configs from the docker config file
// using the registry as the key
func getDockerAuthConfigs() (map[string]registry.AuthConfig, error) {
cfg, err := getDockerConfig()
cfg, err := core.DockerConfig()
if err != nil {
if errors.Is(err, os.ErrNotExist) {
return map[string]registry.AuthConfig{}, nil
Expand Down Expand Up @@ -258,25 +258,3 @@ func getDockerAuthConfigs() (map[string]registry.AuthConfig, error) {

return cfgs, nil
}

// getDockerConfig returns the docker config file. It will internally check, in this particular order:
// 1. the DOCKER_AUTH_CONFIG environment variable, unmarshalling it into a dockercfg.Config
// 2. the DOCKER_CONFIG environment variable, as the path to the config file
// 3. else it will load the default config file, which is ~/.docker/config.json
func getDockerConfig() (*dockercfg.Config, error) {
if env := os.Getenv("DOCKER_AUTH_CONFIG"); env != "" {
var cfg dockercfg.Config
if err := json.Unmarshal([]byte(env), &cfg); err != nil {
return nil, fmt.Errorf("unmarshal DOCKER_AUTH_CONFIG: %w", err)
}

return &cfg, nil
}

cfg, err := dockercfg.LoadDefaultConfig()
if err != nil {
return nil, fmt.Errorf("load default config: %w", err)
}

return &cfg, nil
}
81 changes: 0 additions & 81 deletions docker_auth_test.go
Original file line number Diff line number Diff line change
Expand Up @@ -26,76 +26,6 @@ const (
exampleRegistry = "https://example.com"
)

func Test_getDockerConfig(t *testing.T) {
expectedConfig := &dockercfg.Config{
AuthConfigs: map[string]dockercfg.AuthConfig{
core.IndexDockerIO: {},
exampleRegistry: {},
privateRegistry: {},
},
CredentialsStore: "desktop",
}
t.Run("HOME/valid", func(t *testing.T) {
testDockerConfigHome(t, "testdata")

cfg, err := getDockerConfig()
require.NoError(t, err)
require.Equal(t, expectedConfig, cfg)
})

t.Run("HOME/not-found", func(t *testing.T) {
testDockerConfigHome(t, "testdata", "not-found")

cfg, err := getDockerConfig()
require.ErrorIs(t, err, os.ErrNotExist)
require.Nil(t, cfg)
})

t.Run("HOME/invalid-config", func(t *testing.T) {
testDockerConfigHome(t, "testdata", "invalid-config")

cfg, err := getDockerConfig()
require.ErrorContains(t, err, "json: cannot unmarshal array")
require.Nil(t, cfg)
})

t.Run("DOCKER_AUTH_CONFIG/valid", func(t *testing.T) {
testDockerConfigHome(t, "testdata", "not-found")
t.Setenv("DOCKER_AUTH_CONFIG", dockerConfig)

cfg, err := getDockerConfig()
require.NoError(t, err)
require.Equal(t, expectedConfig, cfg)
})

t.Run("DOCKER_AUTH_CONFIG/invalid-config", func(t *testing.T) {
testDockerConfigHome(t, "testdata", "not-found")
t.Setenv("DOCKER_AUTH_CONFIG", `{"auths": []}`)

cfg, err := getDockerConfig()
require.ErrorContains(t, err, "json: cannot unmarshal array")
require.Nil(t, cfg)
})

t.Run("DOCKER_CONFIG/valid", func(t *testing.T) {
testDockerConfigHome(t, "testdata", "not-found")
t.Setenv("DOCKER_CONFIG", filepath.Join("testdata", ".docker"))

cfg, err := getDockerConfig()
require.NoError(t, err)
require.Equal(t, expectedConfig, cfg)
})

t.Run("DOCKER_CONFIG/invalid-config", func(t *testing.T) {
testDockerConfigHome(t, "testdata", "not-found")
t.Setenv("DOCKER_CONFIG", filepath.Join("testdata", "invalid-config", ".docker"))

cfg, err := getDockerConfig()
require.ErrorContains(t, err, "json: cannot unmarshal array")
require.Nil(t, cfg)
})
}

func TestDockerImageAuth(t *testing.T) {
t.Run("retrieve auth with DOCKER_AUTH_CONFIG env var", func(t *testing.T) {
username, password := "gopher", "secret"
Expand Down Expand Up @@ -297,7 +227,6 @@ func prepareLocalRegistryWithAuth(t *testing.T) string {
// }

genContainerReq := GenericContainerRequest{
ProviderType: providerType,
ContainerRequest: req,
Started: true,
}
Expand All @@ -322,7 +251,6 @@ func prepareLocalRegistryWithAuth(t *testing.T) string {

func prepareRedisImage(ctx context.Context, req ContainerRequest) (Container, error) {
genContainerReq := GenericContainerRequest{
ProviderType: providerType,
ContainerRequest: req,
Started: true,
}
Expand Down Expand Up @@ -474,15 +402,6 @@ func Test_getDockerAuthConfigs(t *testing.T) {

requireValidAuthConfig(t)
})

t.Run("DOCKER_CONFIG/invalid-config", func(t *testing.T) {
testDockerConfigHome(t, "testdata", "not-found")
t.Setenv("DOCKER_CONFIG", filepath.Join("testdata", "invalid-config", ".docker"))

cfg, err := getDockerConfig()
require.ErrorContains(t, err, "json: cannot unmarshal array")
require.Nil(t, cfg)
})
}

// requireValidAuthConfig checks that the given authConfigs map contains the expected keys.
Expand Down
Loading
Loading