2017-10-11 21:27:22 +08:00
using System ;
using System.IO ;
using System.Linq ;
using System.Runtime.InteropServices ;
using System.Security.Cryptography.X509Certificates ;
2018-04-20 08:40:01 -07:00
using System.Threading.Tasks ;
2017-10-11 21:27:22 +08:00
using k8s.Exceptions ;
using k8s.KubeConfigModels ;
2018-04-20 08:40:01 -07:00
2017-10-11 21:27:22 +08:00
namespace k8s
{
public partial class KubernetesClientConfiguration
{
/// <summary>
2017-11-12 03:56:56 +08:00
/// kubeconfig Default Location
2017-10-11 21:27:22 +08:00
/// </summary>
private static readonly string KubeConfigDefaultLocation =
RuntimeInformation . IsOSPlatform ( OSPlatform . Windows )
? Path . Combine ( Environment . GetEnvironmentVariable ( "USERPROFILE" ) , @".kube\config" )
: Path . Combine ( Environment . GetEnvironmentVariable ( "HOME" ) , ".kube/config" ) ;
/// <summary>
2017-11-12 03:56:56 +08:00
/// Gets CurrentContext
/// </summary>
public string CurrentContext { get ; private set ; }
/// <summary>
/// Initializes a new instance of the <see cref="KubernetesClientConfiguration" /> from config file
2017-10-11 21:27:22 +08:00
/// </summary>
/// <param name="masterUrl">kube api server endpoint</param>
2018-04-27 06:13:48 +02:00
/// <param name="kubeconfigPath">Explicit file path to kubeconfig. Set to null to use the default file path</param>
/// <param name="useRelativePaths">When <see langword="true"/>, the paths in the kubeconfig file will be considered to be relative to the directory in which the kubeconfig
/// file is located. When <see langword="false"/>, the paths will be considered to be relative to the current working directory.</param>
2018-04-20 08:40:01 -07:00
public static KubernetesClientConfiguration BuildConfigFromConfigFile ( string kubeconfigPath = null ,
2018-04-27 06:13:48 +02:00
string currentContext = null , string masterUrl = null , bool useRelativePaths = true )
2017-10-11 21:27:22 +08:00
{
2017-11-12 03:56:56 +08:00
return BuildConfigFromConfigFile ( new FileInfo ( kubeconfigPath ? ? KubeConfigDefaultLocation ) , null ,
2018-04-27 06:13:48 +02:00
masterUrl , useRelativePaths ) ;
2017-10-12 16:55:59 +08:00
}
/// <summary>
/// </summary>
/// <param name="kubeconfig">Fileinfo of the kubeconfig, cannot be null</param>
/// <param name="currentContext">override the context in config file, set null if do not want to override</param>
2018-04-27 06:13:48 +02:00
/// <param name="masterUrl">override the kube api server endpoint, set null if do not want to override</param>
/// <param name="useRelativePaths">When <see langword="true"/>, the paths in the kubeconfig file will be considered to be relative to the directory in which the kubeconfig
/// file is located. When <see langword="false"/>, the paths will be considered to be relative to the current working directory.</param>
2017-11-12 03:56:56 +08:00
public static KubernetesClientConfiguration BuildConfigFromConfigFile ( FileInfo kubeconfig ,
2018-04-27 06:13:48 +02:00
string currentContext = null , string masterUrl = null , bool useRelativePaths = true )
2017-10-12 16:55:59 +08:00
{
if ( kubeconfig = = null )
{
throw new NullReferenceException ( nameof ( kubeconfig ) ) ;
}
2018-04-27 06:13:48 +02:00
var k8SConfig = LoadKubeConfig ( kubeconfig , useRelativePaths ) ;
2018-01-31 23:21:14 -02:00
var k8SConfiguration = GetKubernetesClientConfiguration ( currentContext , masterUrl , k8SConfig ) ;
return k8SConfiguration ;
}
/// <summary>
/// </summary>
/// <param name="kubeconfig">Fileinfo of the kubeconfig, cannot be null, whitespaced or empty</param>
/// <param name="currentContext">override the context in config file, set null if do not want to override</param>
/// <param name="masterUrl">overrider kube api server endpoint, set null if do not want to override</param>
public static KubernetesClientConfiguration BuildConfigFromConfigFile ( Stream kubeconfig ,
string currentContext = null , string masterUrl = null )
{
if ( kubeconfig = = null )
{
throw new NullReferenceException ( nameof ( kubeconfig ) ) ;
}
if ( ! kubeconfig . CanSeek )
{
throw new Exception ( "Stream don't support seeking!" ) ;
}
kubeconfig . Position = 0 ;
2018-04-18 21:52:19 -07:00
var k8SConfig = Yaml . LoadFromStreamAsync < K8SConfiguration > ( kubeconfig ) . GetAwaiter ( ) . GetResult ( ) ;
2018-01-31 23:21:14 -02:00
var k8SConfiguration = GetKubernetesClientConfiguration ( currentContext , masterUrl , k8SConfig ) ;
return k8SConfiguration ;
}
private static KubernetesClientConfiguration GetKubernetesClientConfiguration ( string currentContext , string masterUrl , K8SConfiguration k8SConfig )
{
2017-10-11 21:27:22 +08:00
var k8SConfiguration = new KubernetesClientConfiguration ( ) ;
2017-11-12 04:42:37 +08:00
currentContext = currentContext ? ? k8SConfig . CurrentContext ;
// only init context if context if set
if ( currentContext ! = null )
{
2017-12-14 15:22:39 -08:00
k8SConfiguration . InitializeContext ( k8SConfig , currentContext ) ;
}
2017-10-11 21:27:22 +08:00
if ( ! string . IsNullOrWhiteSpace ( masterUrl ) )
{
k8SConfiguration . Host = masterUrl ;
}
2017-11-12 04:42:37 +08:00
if ( string . IsNullOrWhiteSpace ( k8SConfiguration . Host ) )
{
throw new KubeConfigException ( "Cannot infer server host url either from context or masterUrl" ) ;
2017-12-14 15:22:39 -08:00
}
2017-10-11 21:27:22 +08:00
return k8SConfiguration ;
}
/// <summary>
2017-11-12 03:56:56 +08:00
/// Validates and Intializes Client Configuration
2017-10-11 21:27:22 +08:00
/// </summary>
/// <param name="k8SConfig">Kubernetes Configuration</param>
/// <param name="currentContext">Current Context</param>
2017-11-12 04:42:37 +08:00
private void InitializeContext ( K8SConfiguration k8SConfig , string currentContext )
2017-10-11 21:27:22 +08:00
{
// current context
2017-11-12 03:56:56 +08:00
var activeContext =
2017-10-11 21:27:22 +08:00
k8SConfig . Contexts . FirstOrDefault (
c = > c . Name . Equals ( currentContext , StringComparison . OrdinalIgnoreCase ) ) ;
if ( activeContext = = null )
{
throw new KubeConfigException ( $"CurrentContext: {currentContext} not found in contexts in kubeconfig" ) ;
}
2017-11-12 03:56:56 +08:00
CurrentContext = activeContext . Name ;
2017-10-11 21:27:22 +08:00
// cluster
2017-11-12 03:56:56 +08:00
SetClusterDetails ( k8SConfig , activeContext ) ;
// user
SetUserDetails ( k8SConfig , activeContext ) ;
2017-12-14 15:22:39 -08:00
// namespace
Namespace = activeContext . Namespace ;
2017-11-12 03:56:56 +08:00
}
private void SetClusterDetails ( K8SConfiguration k8SConfig , Context activeContext )
{
2017-10-11 21:27:22 +08:00
var clusterDetails =
k8SConfig . Clusters . FirstOrDefault ( c = > c . Name . Equals ( activeContext . ContextDetails . Cluster ,
StringComparison . OrdinalIgnoreCase ) ) ;
2017-11-12 03:56:56 +08:00
2017-10-11 21:27:22 +08:00
if ( clusterDetails ? . ClusterEndpoint = = null )
{
throw new KubeConfigException ( $"Cluster not found for context {activeContext} in kubeconfig" ) ;
}
if ( string . IsNullOrWhiteSpace ( clusterDetails . ClusterEndpoint . Server ) )
{
throw new KubeConfigException ( $"Server not found for current-context {activeContext} in kubeconfig" ) ;
}
2017-11-12 03:56:56 +08:00
Host = clusterDetails . ClusterEndpoint . Server ;
2017-10-11 21:27:22 +08:00
2017-11-12 03:56:56 +08:00
SkipTlsVerify = clusterDetails . ClusterEndpoint . SkipTlsVerify ;
2017-10-11 21:27:22 +08:00
2017-11-12 03:56:56 +08:00
try
2017-10-11 21:27:22 +08:00
{
2017-11-12 03:56:56 +08:00
var uri = new Uri ( Host ) ;
if ( uri . Scheme = = "https" )
{
2017-12-14 15:22:39 -08:00
// check certificate for https
2017-11-12 03:56:56 +08:00
if ( ! clusterDetails . ClusterEndpoint . SkipTlsVerify & &
string . IsNullOrWhiteSpace ( clusterDetails . ClusterEndpoint . CertificateAuthorityData ) & &
string . IsNullOrWhiteSpace ( clusterDetails . ClusterEndpoint . CertificateAuthority ) )
{
throw new KubeConfigException (
$"neither certificate-authority-data nor certificate-authority not found for current-context :{activeContext} in kubeconfig" ) ;
}
if ( ! string . IsNullOrEmpty ( clusterDetails . ClusterEndpoint . CertificateAuthorityData ) )
{
var data = clusterDetails . ClusterEndpoint . CertificateAuthorityData ;
SslCaCert = new X509Certificate2 ( Convert . FromBase64String ( data ) ) ;
}
else if ( ! string . IsNullOrEmpty ( clusterDetails . ClusterEndpoint . CertificateAuthority ) )
{
2018-04-27 06:13:48 +02:00
SslCaCert = new X509Certificate2 ( GetFullPath ( k8SConfig , clusterDetails . ClusterEndpoint . CertificateAuthority ) ) ;
2017-11-12 03:56:56 +08:00
}
}
2017-10-11 21:27:22 +08:00
}
2017-11-12 03:56:56 +08:00
catch ( UriFormatException e )
2017-10-11 21:27:22 +08:00
{
2017-11-12 03:56:56 +08:00
throw new KubeConfigException ( "Bad Server host url" , e ) ;
2017-10-11 21:27:22 +08:00
}
}
private void SetUserDetails ( K8SConfiguration k8SConfig , Context activeContext )
{
2017-11-12 03:56:56 +08:00
if ( string . IsNullOrWhiteSpace ( activeContext . ContextDetails . User ) )
{
return ;
}
2017-10-11 21:27:22 +08:00
var userDetails = k8SConfig . Users . FirstOrDefault ( c = > c . Name . Equals ( activeContext . ContextDetails . User ,
StringComparison . OrdinalIgnoreCase ) ) ;
if ( userDetails = = null )
{
throw new KubeConfigException ( "User not found for context {activeContext.Name} in kubeconfig" ) ;
}
if ( userDetails . UserCredentials = = null )
{
throw new KubeConfigException ( $"User credentials not found for user: {userDetails.Name} in kubeconfig" ) ;
}
var userCredentialsFound = false ;
// Basic and bearer tokens are mutually exclusive
if ( ! string . IsNullOrWhiteSpace ( userDetails . UserCredentials . Token ) )
{
2017-11-12 03:56:56 +08:00
AccessToken = userDetails . UserCredentials . Token ;
2017-10-11 21:27:22 +08:00
userCredentialsFound = true ;
}
else if ( ! string . IsNullOrWhiteSpace ( userDetails . UserCredentials . UserName ) & &
! string . IsNullOrWhiteSpace ( userDetails . UserCredentials . Password ) )
{
2017-11-12 03:56:56 +08:00
Username = userDetails . UserCredentials . UserName ;
Password = userDetails . UserCredentials . Password ;
2017-10-11 21:27:22 +08:00
userCredentialsFound = true ;
}
// Token and cert based auth can co-exist
if ( ! string . IsNullOrWhiteSpace ( userDetails . UserCredentials . ClientCertificateData ) & &
! string . IsNullOrWhiteSpace ( userDetails . UserCredentials . ClientKeyData ) )
{
2017-11-12 03:56:56 +08:00
ClientCertificateData = userDetails . UserCredentials . ClientCertificateData ;
ClientCertificateKeyData = userDetails . UserCredentials . ClientKeyData ;
2017-10-11 21:27:22 +08:00
userCredentialsFound = true ;
}
if ( ! string . IsNullOrWhiteSpace ( userDetails . UserCredentials . ClientCertificate ) & &
! string . IsNullOrWhiteSpace ( userDetails . UserCredentials . ClientKey ) )
{
2018-04-27 06:13:48 +02:00
ClientCertificateFilePath = GetFullPath ( k8SConfig , userDetails . UserCredentials . ClientCertificate ) ;
ClientKeyFilePath = GetFullPath ( k8SConfig , userDetails . UserCredentials . ClientKey ) ;
2017-10-11 21:27:22 +08:00
userCredentialsFound = true ;
}
if ( ! userCredentialsFound )
{
throw new KubeConfigException (
$"User: {userDetails.Name} does not have appropriate auth credentials in kubeconfig" ) ;
}
2018-04-20 08:40:01 -07:00
}
2017-10-11 21:27:22 +08:00
/// <summary>
2018-04-18 21:52:19 -07:00
/// Loads entire Kube Config from default or explicit file path
/// </summary>
2018-04-27 06:13:48 +02:00
/// <param name="kubeconfigPath">Explicit file path to kubeconfig. Set to null to use the default file path</param>
/// <param name="useRelativePaths">When <see langword="true"/>, the paths in the kubeconfig file will be considered to be relative to the directory in which the kubeconfig
/// file is located. When <see langword="false"/>, the paths will be considered to be relative to the current working directory.</param>
/// <returns>Instance of the <see cref="K8SConfiguration"/> class</returns>
public static async Task < K8SConfiguration > LoadKubeConfigAsync ( string kubeconfigPath = null , bool useRelativePaths = true )
2018-04-20 08:40:01 -07:00
{
var fileInfo = new FileInfo ( kubeconfigPath ? ? KubeConfigDefaultLocation ) ;
2018-04-27 06:13:48 +02:00
return await LoadKubeConfigAsync ( fileInfo , useRelativePaths ) ;
2018-04-20 08:40:01 -07:00
}
2018-04-18 21:52:19 -07:00
/// <summary>
/// Loads entire Kube Config from default or explicit file path
/// </summary>
2018-04-27 06:13:48 +02:00
/// <param name="kubeconfigPath">Explicit file path to kubeconfig. Set to null to use the default file path</param>
/// <param name="useRelativePaths">When <see langword="true"/>, the paths in the kubeconfig file will be considered to be relative to the directory in which the kubeconfig
/// file is located. When <see langword="false"/>, the paths will be considered to be relative to the current working directory.</param>
/// <returns>Instance of the <see cref="K8SConfiguration"/> class</returns>
public static K8SConfiguration LoadKubeConfig ( string kubeconfigPath = null , bool useRelativePaths = true )
2018-04-20 08:40:01 -07:00
{
2018-04-27 06:13:48 +02:00
return LoadKubeConfigAsync ( kubeconfigPath , useRelativePaths ) . GetAwaiter ( ) . GetResult ( ) ;
2018-04-20 08:40:01 -07:00
}
2018-04-27 06:13:48 +02:00
2018-04-18 21:52:19 -07:00
// <summary>
2017-11-12 03:56:56 +08:00
/// Loads Kube Config
2017-10-11 21:27:22 +08:00
/// </summary>
2018-04-27 06:13:48 +02:00
/// <param name="kubeconfig">Kube config file contents</param>
/// <param name="useRelativePaths">When <see langword="true"/>, the paths in the kubeconfig file will be considered to be relative to the directory in which the kubeconfig
/// file is located. When <see langword="false"/>, the paths will be considered to be relative to the current working directory.</param>
2017-10-11 21:27:22 +08:00
/// <returns>Instance of the <see cref="K8SConfiguration"/> class</returns>
2018-04-27 06:13:48 +02:00
public static async Task < K8SConfiguration > LoadKubeConfigAsync ( FileInfo kubeconfig , bool useRelativePaths = true )
2017-10-11 21:27:22 +08:00
{
if ( ! kubeconfig . Exists )
{
throw new KubeConfigException ( $"kubeconfig file not found at {kubeconfig.FullName}" ) ;
2018-04-20 08:40:01 -07:00
}
2018-04-18 21:52:19 -07:00
using ( var stream = kubeconfig . OpenRead ( ) )
2018-04-20 08:40:01 -07:00
{
2018-04-27 06:13:48 +02:00
var config = await Yaml . LoadFromStreamAsync < K8SConfiguration > ( stream ) ;
if ( useRelativePaths )
{
config . FileName = kubeconfig . FullName ;
}
return config ;
2017-12-20 09:43:13 -08:00
}
2017-10-11 21:27:22 +08:00
}
2018-01-31 23:21:14 -02:00
/// <summary>
2018-04-18 21:52:19 -07:00
/// Loads Kube Config
2018-01-31 23:21:14 -02:00
/// </summary>
2018-04-27 06:13:48 +02:00
/// <param name="kubeconfig">Kube config file contents</param>
/// <param name="useRelativePaths">When <see langword="true"/>, the paths in the kubeconfig file will be considered to be relative to the directory in which the kubeconfig
/// file is located. When <see langword="false"/>, the paths will be considered to be relative to the current working directory.</param>
2018-01-31 23:21:14 -02:00
/// <returns>Instance of the <see cref="K8SConfiguration"/> class</returns>
2018-04-27 06:13:48 +02:00
public static K8SConfiguration LoadKubeConfig ( FileInfo kubeconfig , bool useRelativePaths = true )
2018-04-20 08:40:01 -07:00
{
2018-04-27 06:13:48 +02:00
return LoadKubeConfigAsync ( kubeconfig , useRelativePaths ) . GetAwaiter ( ) . GetResult ( ) ;
2018-04-20 08:40:01 -07:00
}
2018-04-18 21:52:19 -07:00
// <summary>
/// Loads Kube Config
/// </summary>
/// <param name="kubeconfigStream">Kube config file contents stream</param>
/// <returns>Instance of the <see cref="K8SConfiguration"/> class</returns>
public static async Task < K8SConfiguration > LoadKubeConfigAsync ( Stream kubeconfigStream )
2018-04-20 08:40:01 -07:00
{
2018-04-18 21:52:19 -07:00
return await Yaml . LoadFromStreamAsync < K8SConfiguration > ( kubeconfigStream ) ;
2018-01-31 23:21:14 -02:00
}
/// <summary>
2018-04-18 21:52:19 -07:00
/// Loads Kube Config
2018-01-31 23:21:14 -02:00
/// </summary>
2018-04-18 21:52:19 -07:00
/// <param name="kubeconfig">Kube config file contents stream</param>
2018-01-31 23:21:14 -02:00
/// <returns>Instance of the <see cref="K8SConfiguration"/> class</returns>
2018-04-18 21:52:19 -07:00
public static K8SConfiguration LoadKubeConfig ( Stream kubeconfigStream )
2018-04-20 08:40:01 -07:00
{
2018-04-18 21:52:19 -07:00
return LoadKubeConfigAsync ( kubeconfigStream ) . GetAwaiter ( ) . GetResult ( ) ;
2018-01-31 23:21:14 -02:00
}
2018-04-27 06:13:48 +02:00
/// <summary>
/// Tries to get the full path to a file referenced from the Kubernetes configuration.
/// </summary>
/// <param name="configuration">
/// The Kubernetes configuration.
/// </param>
/// <param name="path">
/// The path to resolve.
/// </param>
/// <returns>
/// When possible a fully qualified path to the file.
/// </returns>
/// <remarks>
/// For example, if the configuration file is at "C:\Users\me\kube.config" and path is "ca.crt",
/// this will return "C:\Users\me\ca.crt". Similarly, if path is "D:\ca.cart", this will return
/// "D:\ca.crt".
/// </remarks>
private static string GetFullPath ( K8SConfiguration configuration , string path )
{
// If we don't have a file name,
if ( string . IsNullOrWhiteSpace ( configuration . FileName ) | | Path . IsPathRooted ( path ) )
{
return path ;
}
else
{
return Path . Combine ( Path . GetDirectoryName ( configuration . FileName ) , path ) ;
}
}
2017-10-11 21:27:22 +08:00
}
2017-11-12 03:56:56 +08:00
}