diff --git a/src/Testcontainers/Builders/DockerRegistryAuthenticationProvider.cs b/src/Testcontainers/Builders/DockerRegistryAuthenticationProvider.cs
index a510858c9..5db612a21 100644
--- a/src/Testcontainers/Builders/DockerRegistryAuthenticationProvider.cs
+++ b/src/Testcontainers/Builders/DockerRegistryAuthenticationProvider.cs
@@ -84,7 +84,11 @@ private IDockerRegistryAuthenticationConfiguration GetUncachedAuthConfig(string
{
authConfig = new IDockerRegistryAuthenticationProvider[]
{
- new CredsHelperProvider(dockerConfigDocument, this.logger), new CredsStoreProvider(dockerConfigDocument, this.logger), new Base64Provider(dockerConfigDocument, this.logger),
+ // environment variable provider goes first as it should overwrite the Docker configuration file when set
+ new EnvironmentVariableBase64Provider(this.logger),
+ new CredsHelperProvider(dockerConfigDocument, this.logger),
+ new CredsStoreProvider(dockerConfigDocument, this.logger),
+ new Base64Provider(dockerConfigDocument, this.logger),
}
.AsParallel()
.Select(authenticationProvider => authenticationProvider.GetAuthConfig(hostname))
diff --git a/src/Testcontainers/Builders/EnvironmentVariableBase64Provider.cs b/src/Testcontainers/Builders/EnvironmentVariableBase64Provider.cs
new file mode 100644
index 000000000..9ffbc8cad
--- /dev/null
+++ b/src/Testcontainers/Builders/EnvironmentVariableBase64Provider.cs
@@ -0,0 +1,90 @@
+namespace DotNet.Testcontainers.Builders
+{
+ using System;
+ using System.Linq;
+ using System.Text;
+ using System.Text.Json;
+ using DotNet.Testcontainers.Configurations;
+ using JetBrains.Annotations;
+ using Microsoft.Extensions.Logging;
+
+ internal sealed class EnvironmentVariableBase64Provider : IDockerRegistryAuthenticationProvider
+ {
+ private readonly JsonElement rootElement;
+ private readonly ILogger logger;
+
+ ///
+ /// Initializes a new instance of the class.
+ ///
+ /// The logger.
+ [PublicAPI]
+ public EnvironmentVariableBase64Provider(ILogger logger)
+ {
+ var environmentVariableValue = Environment.GetEnvironmentVariable("DOCKER_AUTH_CONFIG");
+ if (!string.IsNullOrEmpty(environmentVariableValue))
+ {
+ try
+ {
+ this.rootElement = JsonDocument.Parse(environmentVariableValue).RootElement.TryGetProperty("auths", out var auths) ? auths : default;
+ }
+ catch (JsonException)
+ {
+ // silent
+ }
+ }
+
+ this.logger = logger;
+ }
+
+ public bool IsApplicable(string hostname)
+ {
+#if NETSTANDARD2_1_OR_GREATER
+ return !default(JsonElement).Equals(this.rootElement) && !JsonValueKind.Null.Equals(this.rootElement.ValueKind) && this.rootElement.EnumerateObject().Any(property => property.Name.Contains(hostname, StringComparison.OrdinalIgnoreCase));
+#else
+ return !default(JsonElement).Equals(this.rootElement) && !JsonValueKind.Null.Equals(this.rootElement.ValueKind) && this.rootElement.EnumerateObject().Any(property => property.Name.IndexOf(hostname, StringComparison.OrdinalIgnoreCase) >= 0);
+#endif
+ }
+
+ public IDockerRegistryAuthenticationConfiguration GetAuthConfig(string hostname)
+ {
+ this.logger.SearchingDockerRegistryCredential("EnvironmentVariableAuths");
+
+ if (!this.IsApplicable(hostname))
+ {
+ return null;
+ }
+
+#if NETSTANDARD2_1_OR_GREATER
+ var authProperty = this.rootElement.EnumerateObject().LastOrDefault(property => property.Name.Contains(hostname, StringComparison.OrdinalIgnoreCase));
+#else
+ var authProperty = this.rootElement.EnumerateObject().LastOrDefault(property => property.Name.IndexOf(hostname, StringComparison.OrdinalIgnoreCase) >= 0);
+#endif
+
+ if (JsonValueKind.Undefined.Equals(authProperty.Value.ValueKind))
+ {
+ return null;
+ }
+
+ if (!authProperty.Value.TryGetProperty("auth", out var auth))
+ {
+ return null;
+ }
+
+ if (string.IsNullOrEmpty(auth.GetString()))
+ {
+ return null;
+ }
+
+ var credentialInBytes = Convert.FromBase64String(auth.GetString());
+ var credential = Encoding.UTF8.GetString(credentialInBytes).Split(new[] { ':' }, 2);
+
+ if (credential.Length != 2)
+ {
+ return null;
+ }
+
+ this.logger.DockerRegistryCredentialFound(hostname);
+ return new DockerRegistryAuthenticationConfiguration(authProperty.Name, credential[0], credential[1]);
+ }
+ }
+}
diff --git a/tests/Testcontainers.Tests/Unit/Configurations/DockerRegistryAuthenticationProviderTest.cs b/tests/Testcontainers.Tests/Unit/Configurations/DockerRegistryAuthenticationProviderTest.cs
index ca7fd799b..e8854a822 100644
--- a/tests/Testcontainers.Tests/Unit/Configurations/DockerRegistryAuthenticationProviderTest.cs
+++ b/tests/Testcontainers.Tests/Unit/Configurations/DockerRegistryAuthenticationProviderTest.cs
@@ -104,6 +104,64 @@ public void ShouldGetAuthConfig()
}
}
+ public sealed class EnvironmentVariableBase64ProviderTest
+ {
+ private sealed class EnvironmentVariableScope : IDisposable
+ {
+ public EnvironmentVariableScope(string authConfigValue)
+ {
+ Environment.SetEnvironmentVariable("DOCKER_AUTH_CONFIG", authConfigValue);
+ }
+
+ public void Dispose()
+ {
+ Environment.SetEnvironmentVariable("DOCKER_AUTH_CONFIG", null);
+ }
+ }
+
+ [Theory]
+ [InlineData("{}", false)]
+ [InlineData("{\"auths\":null}", false)]
+ [InlineData("{\"auths\":{}}", false)]
+ [InlineData("{\"auths\":{\"ghcr.io\":{}}}", false)]
+ [InlineData("{\"auths\":{\"" + DockerRegistry + "\":{}}}", true)]
+ [InlineData("{\"auths\":{\"" + DockerRegistry + "\":{\"auth\":null}}}", true)]
+ [InlineData("{\"auths\":{\"" + DockerRegistry + "\":{\"auth\":\"\"}}}", true)]
+ [InlineData("{\"auths\":{\"" + DockerRegistry + "\":{\"auth\":\"dXNlcm5hbWU=\"}}}", true)]
+ public void ShouldGetNull(string jsonAuthConfig, bool isApplicable)
+ {
+ // Given
+ using var scope = new EnvironmentVariableScope(jsonAuthConfig);
+
+ // When
+ var authenticationProvider = new EnvironmentVariableBase64Provider(TestcontainersSettings.Logger);
+ var authConfig = authenticationProvider.GetAuthConfig(DockerRegistry);
+
+ // Then
+ Assert.Equal(isApplicable, authenticationProvider.IsApplicable(DockerRegistry));
+ Assert.Null(authConfig);
+ }
+
+ [Fact]
+ public void ShouldGetAuthConfig()
+ {
+ // Given
+ const string jsonAuthConfig = "{\"auths\":{\"" + DockerRegistry + "\":{\"auth\":\"dXNlcm5hbWU6cGFzc3dvcmQ=\"}}}";
+ using var scope = new EnvironmentVariableScope(jsonAuthConfig);
+
+ // When
+ var authenticationProvider = new EnvironmentVariableBase64Provider(TestcontainersSettings.Logger);
+ var authConfig = authenticationProvider.GetAuthConfig(DockerRegistry);
+
+ // Then
+ Assert.True(authenticationProvider.IsApplicable(DockerRegistry));
+ Assert.NotNull(authConfig);
+ Assert.Equal(DockerRegistry, authConfig.RegistryEndpoint);
+ Assert.Equal("username", authConfig.Username);
+ Assert.Equal("password", authConfig.Password);
+ }
+ }
+
public sealed class CredsStoreProviderTest : SetEnvVarPath
{
[Theory]