-
-
Notifications
You must be signed in to change notification settings - Fork 309
/
Copy pathDockerRegistryAuthenticationProvider.cs
124 lines (108 loc) · 4.91 KB
/
DockerRegistryAuthenticationProvider.cs
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
namespace DotNet.Testcontainers.Builders
{
using System;
using System.Collections.Concurrent;
using System.IO;
using System.Linq;
using System.Text.Json;
using DotNet.Testcontainers.Configurations;
using JetBrains.Annotations;
using Microsoft.Extensions.Logging;
/// <inheritdoc cref="IDockerRegistryAuthenticationProvider" />
internal sealed class DockerRegistryAuthenticationProvider : IDockerRegistryAuthenticationProvider
{
private const string DockerHub = "https://index.docker.io/v1/";
private static readonly ConcurrentDictionary<string, Lazy<IDockerRegistryAuthenticationConfiguration>> Credentials = new ConcurrentDictionary<string, Lazy<IDockerRegistryAuthenticationConfiguration>>();
private static readonly string UserProfileDockerConfigDirectoryPath = Path.Combine(Environment.GetFolderPath(Environment.SpecialFolder.UserProfile), ".docker");
private readonly FileInfo dockerConfigFilePath;
private readonly ILogger logger;
/// <summary>
/// Initializes a new instance of the <see cref="DockerRegistryAuthenticationProvider" /> class.
/// </summary>
/// <param name="logger">The logger.</param>
[PublicAPI]
public DockerRegistryAuthenticationProvider(ILogger logger)
: this(GetDefaultDockerConfigFilePath(), logger)
{
}
/// <summary>
/// Initializes a new instance of the <see cref="DockerRegistryAuthenticationProvider" /> class.
/// </summary>
/// <param name="dockerConfigFilePath">The Docker config file path.</param>
/// <param name="logger">The logger.</param>
[PublicAPI]
public DockerRegistryAuthenticationProvider(string dockerConfigFilePath, ILogger logger)
: this(new FileInfo(dockerConfigFilePath), logger)
{
}
/// <summary>
/// Initializes a new instance of the <see cref="DockerRegistryAuthenticationProvider" /> class.
/// </summary>
/// <param name="dockerConfigFilePath">The Docker config file path.</param>
/// <param name="logger">The logger.</param>
[PublicAPI]
public DockerRegistryAuthenticationProvider(FileInfo dockerConfigFilePath, ILogger logger)
{
this.dockerConfigFilePath = dockerConfigFilePath;
this.logger = logger;
}
/// <inheritdoc />
public bool IsApplicable(string hostname)
{
return true;
}
/// <inheritdoc />
public IDockerRegistryAuthenticationConfiguration GetAuthConfig(string hostname)
{
var lazyAuthConfig = Credentials.GetOrAdd(hostname ?? DockerHub, key => new Lazy<IDockerRegistryAuthenticationConfiguration>(() => this.GetUncachedAuthConfig(key)));
return lazyAuthConfig.Value;
}
private static string GetDefaultDockerConfigFilePath()
{
var dockerConfigDirectoryPath = PropertiesFileConfiguration.Instance.GetDockerConfig() ?? EnvironmentConfiguration.Instance.GetDockerConfig() ?? UserProfileDockerConfigDirectoryPath;
return Path.Combine(dockerConfigDirectoryPath, "config.json");
}
private static JsonDocument GetDefaultDockerAuthConfig()
{
return PropertiesFileConfiguration.Instance.GetDockerAuthConfig() ?? EnvironmentConfiguration.Instance.GetDockerAuthConfig() ?? JsonDocument.Parse("{}");
}
private IDockerRegistryAuthenticationConfiguration GetUncachedAuthConfig(string hostname)
{
using (var dockerAuthConfigJsonDocument = GetDefaultDockerAuthConfig())
{
IDockerRegistryAuthenticationConfiguration authConfig;
if (this.dockerConfigFilePath.Exists)
{
using (var dockerConfigFileStream = new FileStream(this.dockerConfigFilePath.FullName, FileMode.Open, FileAccess.Read))
{
using (var dockerConfigJsonDocument = JsonDocument.Parse(dockerConfigFileStream))
{
authConfig = new IDockerRegistryAuthenticationProvider[]
{
new CredsHelperProvider(dockerConfigJsonDocument, this.logger),
new CredsStoreProvider(dockerConfigJsonDocument, this.logger),
new Base64Provider(dockerConfigJsonDocument, this.logger),
new Base64Provider(dockerAuthConfigJsonDocument, this.logger),
}
.AsParallel()
.Select(authenticationProvider => authenticationProvider.GetAuthConfig(hostname))
.FirstOrDefault(authenticationProvider => authenticationProvider != null);
}
}
}
else
{
this.logger.DockerConfigFileNotFound(this.dockerConfigFilePath.FullName);
IDockerRegistryAuthenticationProvider authConfigProvider = new Base64Provider(dockerAuthConfigJsonDocument, this.logger);
authConfig = authConfigProvider.GetAuthConfig(hostname);
}
if (authConfig != null)
{
return authConfig;
}
this.logger.DockerRegistryCredentialNotFound(hostname);
return default(DockerRegistryAuthenticationConfiguration);
}
}
}
}