2022-03-22 16:03:15 +01:00
|
|
|
package k8sapi
|
|
|
|
|
|
|
|
import (
|
2022-05-20 17:30:37 +02:00
|
|
|
"path/filepath"
|
|
|
|
|
2022-06-29 15:26:29 +02:00
|
|
|
"github.com/edgelesssys/constellation/bootstrapper/internal/kubernetes/k8sapi/resources"
|
2022-05-05 08:48:56 +02:00
|
|
|
"github.com/edgelesssys/constellation/internal/constants"
|
2022-05-20 17:30:37 +02:00
|
|
|
corev1 "k8s.io/api/core/v1"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
2022-03-22 16:03:15 +01:00
|
|
|
kubeletconf "k8s.io/kubelet/config/v1beta1"
|
|
|
|
kubeadm "k8s.io/kubernetes/cmd/kubeadm/app/apis/kubeadm/v1beta3"
|
|
|
|
)
|
|
|
|
|
|
|
|
// Uses types defined here: https://kubernetes.io/docs/reference/config-api/kubeadm-config.v1beta3/
|
|
|
|
// Slimmed down to the fields we require
|
|
|
|
|
|
|
|
const (
|
2022-05-20 17:30:37 +02:00
|
|
|
bindPort = 6443
|
|
|
|
auditLogDir = "/var/log/kubernetes/audit/"
|
|
|
|
auditLogFile = "audit.log"
|
|
|
|
auditPolicyPath = "/etc/kubernetes/audit-policy.yaml"
|
2022-03-22 16:03:15 +01:00
|
|
|
)
|
|
|
|
|
|
|
|
type CoreOSConfiguration struct{}
|
|
|
|
|
2022-04-27 16:37:05 +02:00
|
|
|
func (c *CoreOSConfiguration) InitConfiguration(externalCloudProvider bool) KubeadmInitYAML {
|
|
|
|
var cloudProvider string
|
|
|
|
if externalCloudProvider {
|
|
|
|
cloudProvider = "external"
|
|
|
|
}
|
2022-03-22 16:03:15 +01:00
|
|
|
return KubeadmInitYAML{
|
|
|
|
InitConfiguration: kubeadm.InitConfiguration{
|
2022-05-20 17:30:37 +02:00
|
|
|
TypeMeta: metav1.TypeMeta{
|
2022-03-22 16:03:15 +01:00
|
|
|
APIVersion: kubeadm.SchemeGroupVersion.String(),
|
|
|
|
Kind: "InitConfiguration",
|
|
|
|
},
|
|
|
|
NodeRegistration: kubeadm.NodeRegistrationOptions{
|
2022-04-01 10:05:46 +02:00
|
|
|
CRISocket: "/run/containerd/containerd.sock",
|
2022-03-22 16:03:15 +01:00
|
|
|
KubeletExtraArgs: map[string]string{
|
2022-04-27 16:37:05 +02:00
|
|
|
"cloud-provider": cloudProvider,
|
2022-03-22 16:03:15 +01:00
|
|
|
"network-plugin": "cni",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
// AdvertiseAddress will be overwritten later
|
|
|
|
LocalAPIEndpoint: kubeadm.APIEndpoint{
|
|
|
|
BindPort: bindPort,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
ClusterConfiguration: kubeadm.ClusterConfiguration{
|
2022-05-20 17:30:37 +02:00
|
|
|
TypeMeta: metav1.TypeMeta{
|
2022-03-22 16:03:15 +01:00
|
|
|
Kind: "ClusterConfiguration",
|
|
|
|
APIVersion: kubeadm.SchemeGroupVersion.String(),
|
|
|
|
},
|
2022-05-05 08:48:56 +02:00
|
|
|
KubernetesVersion: constants.KubernetesVersion,
|
2022-03-22 16:03:15 +01:00
|
|
|
// necessary to be able to access the kubeapi server through localhost
|
|
|
|
APIServer: kubeadm.APIServer{
|
2022-05-09 16:12:15 +02:00
|
|
|
ControlPlaneComponent: kubeadm.ControlPlaneComponent{
|
|
|
|
ExtraArgs: map[string]string{
|
2022-05-20 17:30:37 +02:00
|
|
|
"audit-policy-file": auditPolicyPath,
|
|
|
|
"audit-log-path": filepath.Join(auditLogDir, auditLogFile), // CIS benchmark
|
|
|
|
"audit-log-maxage": "30", // CIS benchmark - Default value of Rancher
|
|
|
|
"audit-log-maxbackup": "10", // CIS benchmark - Default value of Rancher
|
|
|
|
"audit-log-maxsize": "100", // CIS benchmark - Default value of Rancher
|
|
|
|
"profiling": "false", // CIS benchmark
|
2022-05-09 16:12:15 +02:00
|
|
|
"tls-cipher-suites": "TLS_AES_128_GCM_SHA256,TLS_AES_256_GCM_SHA384,TLS_CHACHA20_POLY1305_SHA256," +
|
|
|
|
"TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA,TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256," +
|
|
|
|
"TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA,TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384," +
|
|
|
|
"TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305,TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256," +
|
|
|
|
"TLS_ECDHE_RSA_WITH_3DES_EDE_CBC_SHA,TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA,TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256," +
|
|
|
|
"TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA,TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384,TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305," +
|
|
|
|
"TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256,TLS_RSA_WITH_3DES_EDE_CBC_SHA,TLS_RSA_WITH_AES_128_CBC_SHA," +
|
|
|
|
"TLS_RSA_WITH_AES_128_GCM_SHA256,TLS_RSA_WITH_AES_256_CBC_SHA,TLS_RSA_WITH_AES_256_GCM_SHA384", // CIS benchmark
|
|
|
|
},
|
2022-05-20 17:30:37 +02:00
|
|
|
ExtraVolumes: []kubeadm.HostPathMount{
|
|
|
|
{
|
|
|
|
Name: "audit-log",
|
|
|
|
HostPath: auditLogDir,
|
|
|
|
MountPath: auditLogDir,
|
|
|
|
ReadOnly: false,
|
|
|
|
PathType: corev1.HostPathDirectoryOrCreate,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "audit",
|
|
|
|
HostPath: auditPolicyPath,
|
|
|
|
MountPath: auditPolicyPath,
|
|
|
|
ReadOnly: true,
|
|
|
|
PathType: corev1.HostPathFile,
|
|
|
|
},
|
|
|
|
},
|
2022-05-09 16:12:15 +02:00
|
|
|
},
|
2022-07-05 14:14:11 +02:00
|
|
|
CertSANs: []string{"127.0.0.1"},
|
2022-03-22 16:03:15 +01:00
|
|
|
},
|
|
|
|
ControllerManager: kubeadm.ControlPlaneComponent{
|
|
|
|
ExtraArgs: map[string]string{
|
2022-05-09 16:12:15 +02:00
|
|
|
"flex-volume-plugin-dir": "/opt/libexec/kubernetes/kubelet-plugins/volume/exec/",
|
|
|
|
"cloud-provider": cloudProvider,
|
|
|
|
"configure-cloud-routes": "false",
|
|
|
|
"profiling": "false", // CIS benchmark
|
|
|
|
"terminated-pod-gc-threshold": "1000", // CIS benchmark - Default value of Rancher
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Scheduler: kubeadm.ControlPlaneComponent{
|
|
|
|
ExtraArgs: map[string]string{
|
|
|
|
"profiling": "false",
|
2022-03-22 16:03:15 +01:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
// warning: this config is applied to every node in the cluster!
|
|
|
|
KubeletConfiguration: kubeletconf.KubeletConfiguration{
|
2022-05-10 07:50:28 +02:00
|
|
|
ProtectKernelDefaults: true, // CIS benchmark
|
|
|
|
TLSCipherSuites: []string{
|
|
|
|
"TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256",
|
|
|
|
"TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256",
|
|
|
|
"TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305",
|
|
|
|
"TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384",
|
|
|
|
"TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305",
|
|
|
|
"TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384",
|
|
|
|
"TLS_RSA_WITH_AES_256_GCM_SHA384",
|
|
|
|
"TLS_RSA_WITH_AES_128_GCM_SHA256",
|
|
|
|
}, // CIS benchmark
|
2022-05-20 17:30:37 +02:00
|
|
|
TypeMeta: metav1.TypeMeta{
|
2022-03-22 16:03:15 +01:00
|
|
|
APIVersion: kubeletconf.SchemeGroupVersion.String(),
|
|
|
|
Kind: "KubeletConfiguration",
|
|
|
|
},
|
2022-05-24 10:04:42 +02:00
|
|
|
RegisterWithTaints: []corev1.Taint{
|
|
|
|
{
|
|
|
|
Key: "node.cloudprovider.kubernetes.io/uninitialized",
|
|
|
|
Value: "true",
|
|
|
|
Effect: corev1.TaintEffectPreferNoSchedule,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Key: "node.cilium.io/agent-not-ready",
|
|
|
|
Value: "true",
|
|
|
|
Effect: corev1.TaintEffectPreferNoSchedule,
|
|
|
|
},
|
|
|
|
},
|
2022-03-22 16:03:15 +01:00
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-27 16:37:05 +02:00
|
|
|
func (c *CoreOSConfiguration) JoinConfiguration(externalCloudProvider bool) KubeadmJoinYAML {
|
|
|
|
var cloudProvider string
|
|
|
|
if externalCloudProvider {
|
|
|
|
cloudProvider = "external"
|
|
|
|
}
|
2022-03-22 16:03:15 +01:00
|
|
|
return KubeadmJoinYAML{
|
|
|
|
JoinConfiguration: kubeadm.JoinConfiguration{
|
2022-05-20 17:30:37 +02:00
|
|
|
TypeMeta: metav1.TypeMeta{
|
2022-03-22 16:03:15 +01:00
|
|
|
APIVersion: kubeadm.SchemeGroupVersion.String(),
|
|
|
|
Kind: "JoinConfiguration",
|
|
|
|
},
|
|
|
|
NodeRegistration: kubeadm.NodeRegistrationOptions{
|
2022-04-01 10:05:46 +02:00
|
|
|
CRISocket: "/run/containerd/containerd.sock",
|
2022-03-22 16:03:15 +01:00
|
|
|
KubeletExtraArgs: map[string]string{
|
2022-04-27 16:37:05 +02:00
|
|
|
"cloud-provider": cloudProvider,
|
2022-03-22 16:03:15 +01:00
|
|
|
},
|
|
|
|
},
|
|
|
|
Discovery: kubeadm.Discovery{
|
|
|
|
BootstrapToken: &kubeadm.BootstrapTokenDiscovery{},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
KubeletConfiguration: kubeletconf.KubeletConfiguration{
|
2022-05-20 17:30:37 +02:00
|
|
|
TypeMeta: metav1.TypeMeta{
|
2022-03-22 16:03:15 +01:00
|
|
|
APIVersion: kubeletconf.SchemeGroupVersion.String(),
|
|
|
|
Kind: "KubeletConfiguration",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
type KubeadmJoinYAML struct {
|
|
|
|
JoinConfiguration kubeadm.JoinConfiguration
|
|
|
|
KubeletConfiguration kubeletconf.KubeletConfiguration
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmJoinYAML) SetNodeName(nodeName string) {
|
|
|
|
k.JoinConfiguration.NodeRegistration.Name = nodeName
|
|
|
|
}
|
|
|
|
|
2022-07-08 10:59:59 +02:00
|
|
|
func (k *KubeadmJoinYAML) SetAPIServerEndpoint(apiServerEndpoint string) {
|
2022-03-22 16:03:15 +01:00
|
|
|
k.JoinConfiguration.Discovery.BootstrapToken.APIServerEndpoint = apiServerEndpoint
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmJoinYAML) SetToken(token string) {
|
|
|
|
k.JoinConfiguration.Discovery.BootstrapToken.Token = token
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmJoinYAML) AppendDiscoveryTokenCaCertHash(discoveryTokenCaCertHash string) {
|
|
|
|
k.JoinConfiguration.Discovery.BootstrapToken.CACertHashes = append(k.JoinConfiguration.Discovery.BootstrapToken.CACertHashes, discoveryTokenCaCertHash)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmJoinYAML) SetNodeIP(nodeIP string) {
|
|
|
|
if k.JoinConfiguration.NodeRegistration.KubeletExtraArgs == nil {
|
|
|
|
k.JoinConfiguration.NodeRegistration.KubeletExtraArgs = map[string]string{"node-ip": nodeIP}
|
|
|
|
} else {
|
|
|
|
k.JoinConfiguration.NodeRegistration.KubeletExtraArgs["node-ip"] = nodeIP
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmJoinYAML) SetProviderID(providerID string) {
|
|
|
|
k.KubeletConfiguration.ProviderID = providerID
|
|
|
|
}
|
|
|
|
|
2022-07-11 13:29:22 +02:00
|
|
|
func (k *KubeadmJoinYAML) SetControlPlane(advertiseAddress string) {
|
2022-04-25 17:24:48 +02:00
|
|
|
k.JoinConfiguration.ControlPlane = &kubeadm.JoinControlPlane{
|
|
|
|
LocalAPIEndpoint: kubeadm.APIEndpoint{
|
|
|
|
AdvertiseAddress: advertiseAddress,
|
|
|
|
BindPort: 6443,
|
|
|
|
},
|
|
|
|
}
|
2022-07-11 13:29:22 +02:00
|
|
|
k.JoinConfiguration.SkipPhases = []string{"control-plane-prepare/download-certs"}
|
2022-04-25 17:24:48 +02:00
|
|
|
}
|
|
|
|
|
2022-03-22 16:03:15 +01:00
|
|
|
func (k *KubeadmJoinYAML) Marshal() ([]byte, error) {
|
|
|
|
return resources.MarshalK8SResources(k)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmJoinYAML) Unmarshal(yamlData []byte) (KubeadmJoinYAML, error) {
|
|
|
|
var tmp KubeadmJoinYAML
|
|
|
|
return tmp, resources.UnmarshalK8SResources(yamlData, &tmp)
|
|
|
|
}
|
|
|
|
|
|
|
|
type KubeadmInitYAML struct {
|
|
|
|
InitConfiguration kubeadm.InitConfiguration
|
|
|
|
ClusterConfiguration kubeadm.ClusterConfiguration
|
|
|
|
KubeletConfiguration kubeletconf.KubeletConfiguration
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmInitYAML) SetNodeName(nodeName string) {
|
|
|
|
k.InitConfiguration.NodeRegistration.Name = nodeName
|
|
|
|
}
|
|
|
|
|
2022-05-24 10:04:42 +02:00
|
|
|
// SetCertSANs sets the SANs for the certificate.
|
|
|
|
func (k *KubeadmInitYAML) SetCertSANs(certSANs []string) {
|
|
|
|
for _, certSAN := range certSANs {
|
|
|
|
if certSAN == "" {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
k.ClusterConfiguration.APIServer.CertSANs = append(k.ClusterConfiguration.APIServer.CertSANs, certSAN)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-08 10:59:59 +02:00
|
|
|
func (k *KubeadmInitYAML) SetAPIServerAdvertiseAddress(apiServerAdvertiseAddress string) {
|
2022-03-22 16:03:15 +01:00
|
|
|
k.InitConfiguration.LocalAPIEndpoint.AdvertiseAddress = apiServerAdvertiseAddress
|
|
|
|
}
|
|
|
|
|
2022-05-24 10:04:42 +02:00
|
|
|
// SetControlPlaneEndpoint sets the control plane endpoint if controlPlaneEndpoint is not empty.
|
|
|
|
func (k *KubeadmInitYAML) SetControlPlaneEndpoint(controlPlaneEndpoint string) {
|
|
|
|
if controlPlaneEndpoint != "" {
|
|
|
|
k.ClusterConfiguration.ControlPlaneEndpoint = controlPlaneEndpoint
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-22 16:03:15 +01:00
|
|
|
func (k *KubeadmInitYAML) SetServiceCIDR(serviceCIDR string) {
|
|
|
|
k.ClusterConfiguration.Networking.ServiceSubnet = serviceCIDR
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmInitYAML) SetPodNetworkCIDR(podNetworkCIDR string) {
|
|
|
|
k.ClusterConfiguration.Networking.PodSubnet = podNetworkCIDR
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmInitYAML) SetServiceDNSDomain(serviceDNSDomain string) {
|
|
|
|
k.ClusterConfiguration.Networking.DNSDomain = serviceDNSDomain
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmInitYAML) SetNodeIP(nodeIP string) {
|
|
|
|
if k.InitConfiguration.NodeRegistration.KubeletExtraArgs == nil {
|
|
|
|
k.InitConfiguration.NodeRegistration.KubeletExtraArgs = map[string]string{"node-ip": nodeIP}
|
|
|
|
} else {
|
|
|
|
k.InitConfiguration.NodeRegistration.KubeletExtraArgs["node-ip"] = nodeIP
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmInitYAML) SetProviderID(providerID string) {
|
|
|
|
if k.InitConfiguration.NodeRegistration.KubeletExtraArgs == nil {
|
|
|
|
k.InitConfiguration.NodeRegistration.KubeletExtraArgs = map[string]string{"provider-id": providerID}
|
|
|
|
} else {
|
|
|
|
k.InitConfiguration.NodeRegistration.KubeletExtraArgs["provider-id"] = providerID
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmInitYAML) Marshal() ([]byte, error) {
|
|
|
|
return resources.MarshalK8SResources(k)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (k *KubeadmInitYAML) Unmarshal(yamlData []byte) (KubeadmInitYAML, error) {
|
|
|
|
var tmp KubeadmInitYAML
|
|
|
|
return tmp, resources.UnmarshalK8SResources(yamlData, &tmp)
|
|
|
|
}
|