2019-01-12 04:58:27 +00:00
|
|
|
/*
|
|
|
|
Copyright 2017 The Kubernetes Authors.
|
|
|
|
|
|
|
|
Licensed under the Apache License, Version 2.0 (the "License");
|
|
|
|
you may not use this file except in compliance with the License.
|
|
|
|
You may obtain a copy of the License at
|
|
|
|
|
|
|
|
http://www.apache.org/licenses/LICENSE-2.0
|
|
|
|
|
|
|
|
Unless required by applicable law or agreed to in writing, software
|
|
|
|
distributed under the License is distributed on an "AS IS" BASIS,
|
|
|
|
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
|
|
|
See the License for the specific language governing permissions and
|
|
|
|
limitations under the License.
|
|
|
|
*/
|
|
|
|
|
|
|
|
package certificate
|
|
|
|
|
|
|
|
import (
|
2019-04-07 17:07:55 +00:00
|
|
|
"crypto/tls"
|
2019-01-12 04:58:27 +00:00
|
|
|
"crypto/x509"
|
|
|
|
"crypto/x509/pkix"
|
|
|
|
"fmt"
|
2020-03-26 21:07:15 +00:00
|
|
|
"math"
|
2019-01-12 04:58:27 +00:00
|
|
|
"net"
|
|
|
|
"sort"
|
2020-03-26 21:07:15 +00:00
|
|
|
"time"
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2020-08-10 17:43:49 +00:00
|
|
|
certificates "k8s.io/api/certificates/v1"
|
2019-12-12 01:27:03 +00:00
|
|
|
v1 "k8s.io/api/core/v1"
|
2019-01-12 04:58:27 +00:00
|
|
|
"k8s.io/apimachinery/pkg/types"
|
|
|
|
clientset "k8s.io/client-go/kubernetes"
|
|
|
|
"k8s.io/client-go/util/certificate"
|
2019-09-27 21:51:53 +00:00
|
|
|
compbasemetrics "k8s.io/component-base/metrics"
|
|
|
|
"k8s.io/component-base/metrics/legacyregistry"
|
2019-01-12 04:58:27 +00:00
|
|
|
kubeletconfig "k8s.io/kubernetes/pkg/kubelet/apis/config"
|
|
|
|
"k8s.io/kubernetes/pkg/kubelet/metrics"
|
|
|
|
)
|
|
|
|
|
|
|
|
// NewKubeletServerCertificateManager creates a certificate manager for the kubelet when retrieving a server certificate
|
|
|
|
// or returns an error.
|
|
|
|
func NewKubeletServerCertificateManager(kubeClient clientset.Interface, kubeCfg *kubeletconfig.KubeletConfiguration, nodeName types.NodeName, getAddresses func() []v1.NodeAddress, certDirectory string) (certificate.Manager, error) {
|
2020-08-10 17:43:49 +00:00
|
|
|
var clientsetFn certificate.ClientsetFunc
|
|
|
|
if kubeClient != nil {
|
|
|
|
clientsetFn = func(current *tls.Certificate) (clientset.Interface, error) {
|
|
|
|
return kubeClient, nil
|
|
|
|
}
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
certificateStore, err := certificate.NewFileStore(
|
|
|
|
"kubelet-server",
|
|
|
|
certDirectory,
|
|
|
|
certDirectory,
|
|
|
|
kubeCfg.TLSCertFile,
|
|
|
|
kubeCfg.TLSPrivateKeyFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to initialize server certificate store: %v", err)
|
|
|
|
}
|
2020-03-26 21:07:15 +00:00
|
|
|
var certificateRenewFailure = compbasemetrics.NewCounter(
|
|
|
|
&compbasemetrics.CounterOpts{
|
2019-12-12 01:27:03 +00:00
|
|
|
Subsystem: metrics.KubeletSubsystem,
|
2020-03-26 21:07:15 +00:00
|
|
|
Name: "server_expiration_renew_errors",
|
|
|
|
Help: "Counter of certificate renewal errors.",
|
2019-09-27 21:51:53 +00:00
|
|
|
StabilityLevel: compbasemetrics.ALPHA,
|
2019-01-12 04:58:27 +00:00
|
|
|
},
|
|
|
|
)
|
2020-03-26 21:07:15 +00:00
|
|
|
legacyregistry.MustRegister(certificateRenewFailure)
|
2019-01-12 04:58:27 +00:00
|
|
|
|
2019-12-12 01:27:03 +00:00
|
|
|
certificateRotationAge := compbasemetrics.NewHistogram(
|
|
|
|
&compbasemetrics.HistogramOpts{
|
|
|
|
Subsystem: metrics.KubeletSubsystem,
|
|
|
|
Name: "certificate_manager_server_rotation_seconds",
|
|
|
|
Help: "Histogram of the number of seconds the previous certificate lived before being rotated.",
|
|
|
|
Buckets: []float64{
|
|
|
|
60, // 1 minute
|
|
|
|
3600, // 1 hour
|
|
|
|
14400, // 4 hours
|
|
|
|
86400, // 1 day
|
|
|
|
604800, // 1 week
|
|
|
|
2592000, // 1 month
|
|
|
|
7776000, // 3 months
|
|
|
|
15552000, // 6 months
|
|
|
|
31104000, // 1 year
|
|
|
|
124416000, // 4 years
|
|
|
|
},
|
|
|
|
StabilityLevel: compbasemetrics.ALPHA,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
legacyregistry.MustRegister(certificateRotationAge)
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
getTemplate := func() *x509.CertificateRequest {
|
|
|
|
hostnames, ips := addressesToHostnamesAndIPs(getAddresses())
|
|
|
|
// don't return a template if we have no addresses to request for
|
|
|
|
if len(hostnames) == 0 && len(ips) == 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
return &x509.CertificateRequest{
|
|
|
|
Subject: pkix.Name{
|
|
|
|
CommonName: fmt.Sprintf("system:node:%s", nodeName),
|
|
|
|
Organization: []string{"system:nodes"},
|
|
|
|
},
|
|
|
|
DNSNames: hostnames,
|
|
|
|
IPAddresses: ips,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
m, err := certificate.NewManager(&certificate.Config{
|
2020-08-10 17:43:49 +00:00
|
|
|
ClientsetFn: clientsetFn,
|
2019-04-07 17:07:55 +00:00
|
|
|
GetTemplate: getTemplate,
|
2020-03-26 21:07:15 +00:00
|
|
|
SignerName: certificates.KubeletServingSignerName,
|
2019-01-12 04:58:27 +00:00
|
|
|
Usages: []certificates.KeyUsage{
|
|
|
|
// https://tools.ietf.org/html/rfc5280#section-4.2.1.3
|
|
|
|
//
|
|
|
|
// Digital signature allows the certificate to be used to verify
|
|
|
|
// digital signatures used during TLS negotiation.
|
|
|
|
certificates.UsageDigitalSignature,
|
|
|
|
// KeyEncipherment allows the cert/key pair to be used to encrypt
|
|
|
|
// keys, including the symmetric keys negotiated during TLS setup
|
|
|
|
// and used for data transfer.
|
|
|
|
certificates.UsageKeyEncipherment,
|
|
|
|
// ServerAuth allows the cert to be used by a TLS server to
|
|
|
|
// authenticate itself to a TLS client.
|
|
|
|
certificates.UsageServerAuth,
|
|
|
|
},
|
2020-03-26 21:07:15 +00:00
|
|
|
CertificateStore: certificateStore,
|
|
|
|
CertificateRotation: certificateRotationAge,
|
|
|
|
CertificateRenewFailure: certificateRenewFailure,
|
2019-01-12 04:58:27 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to initialize server certificate manager: %v", err)
|
|
|
|
}
|
2020-03-26 21:07:15 +00:00
|
|
|
legacyregistry.RawMustRegister(compbasemetrics.NewGaugeFunc(
|
|
|
|
compbasemetrics.GaugeOpts{
|
|
|
|
Subsystem: metrics.KubeletSubsystem,
|
|
|
|
Name: "certificate_manager_server_ttl_seconds",
|
|
|
|
Help: "Gauge of the shortest TTL (time-to-live) of " +
|
|
|
|
"the Kubelet's serving certificate. The value is in seconds " +
|
|
|
|
"until certificate expiry (negative if already expired). If " +
|
|
|
|
"serving certificate is invalid or unused, the value will " +
|
|
|
|
"be +INF.",
|
|
|
|
StabilityLevel: compbasemetrics.ALPHA,
|
|
|
|
},
|
|
|
|
func() float64 {
|
|
|
|
if c := m.Current(); c != nil && c.Leaf != nil {
|
2020-08-10 17:43:49 +00:00
|
|
|
return math.Trunc(c.Leaf.NotAfter.Sub(time.Now()).Seconds())
|
2020-03-26 21:07:15 +00:00
|
|
|
}
|
|
|
|
return math.Inf(1)
|
|
|
|
},
|
|
|
|
))
|
2019-01-12 04:58:27 +00:00
|
|
|
return m, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func addressesToHostnamesAndIPs(addresses []v1.NodeAddress) (dnsNames []string, ips []net.IP) {
|
|
|
|
seenDNSNames := map[string]bool{}
|
|
|
|
seenIPs := map[string]bool{}
|
|
|
|
for _, address := range addresses {
|
|
|
|
if len(address.Address) == 0 {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
switch address.Type {
|
|
|
|
case v1.NodeHostName:
|
|
|
|
if ip := net.ParseIP(address.Address); ip != nil {
|
|
|
|
seenIPs[address.Address] = true
|
|
|
|
} else {
|
|
|
|
seenDNSNames[address.Address] = true
|
|
|
|
}
|
|
|
|
case v1.NodeExternalIP, v1.NodeInternalIP:
|
|
|
|
if ip := net.ParseIP(address.Address); ip != nil {
|
|
|
|
seenIPs[address.Address] = true
|
|
|
|
}
|
|
|
|
case v1.NodeExternalDNS, v1.NodeInternalDNS:
|
|
|
|
seenDNSNames[address.Address] = true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
for dnsName := range seenDNSNames {
|
|
|
|
dnsNames = append(dnsNames, dnsName)
|
|
|
|
}
|
|
|
|
for ip := range seenIPs {
|
|
|
|
ips = append(ips, net.ParseIP(ip))
|
|
|
|
}
|
|
|
|
|
|
|
|
// return in stable order
|
|
|
|
sort.Strings(dnsNames)
|
|
|
|
sort.Slice(ips, func(i, j int) bool { return ips[i].String() < ips[j].String() })
|
|
|
|
|
|
|
|
return dnsNames, ips
|
|
|
|
}
|
|
|
|
|
|
|
|
// NewKubeletClientCertificateManager sets up a certificate manager without a
|
2019-04-07 17:07:55 +00:00
|
|
|
// client that can be used to sign new certificates (or rotate). If a CSR
|
|
|
|
// client is set later, it may begin rotating/renewing the client cert.
|
|
|
|
func NewKubeletClientCertificateManager(
|
|
|
|
certDirectory string,
|
|
|
|
nodeName types.NodeName,
|
|
|
|
bootstrapCertData []byte,
|
|
|
|
bootstrapKeyData []byte,
|
|
|
|
certFile string,
|
|
|
|
keyFile string,
|
2020-08-10 17:43:49 +00:00
|
|
|
clientsetFn certificate.ClientsetFunc,
|
2019-04-07 17:07:55 +00:00
|
|
|
) (certificate.Manager, error) {
|
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
certificateStore, err := certificate.NewFileStore(
|
|
|
|
"kubelet-client",
|
|
|
|
certDirectory,
|
|
|
|
certDirectory,
|
|
|
|
certFile,
|
|
|
|
keyFile)
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to initialize client certificate store: %v", err)
|
|
|
|
}
|
2020-03-26 21:07:15 +00:00
|
|
|
var certificateRenewFailure = compbasemetrics.NewCounter(
|
|
|
|
&compbasemetrics.CounterOpts{
|
|
|
|
Namespace: metrics.KubeletSubsystem,
|
|
|
|
Subsystem: "certificate_manager",
|
|
|
|
Name: "client_expiration_renew_errors",
|
|
|
|
Help: "Counter of certificate renewal errors.",
|
|
|
|
StabilityLevel: compbasemetrics.ALPHA,
|
|
|
|
},
|
|
|
|
)
|
|
|
|
legacyregistry.Register(certificateRenewFailure)
|
2019-01-12 04:58:27 +00:00
|
|
|
|
|
|
|
m, err := certificate.NewManager(&certificate.Config{
|
2020-08-10 17:43:49 +00:00
|
|
|
ClientsetFn: clientsetFn,
|
2019-01-12 04:58:27 +00:00
|
|
|
Template: &x509.CertificateRequest{
|
|
|
|
Subject: pkix.Name{
|
|
|
|
CommonName: fmt.Sprintf("system:node:%s", nodeName),
|
|
|
|
Organization: []string{"system:nodes"},
|
|
|
|
},
|
|
|
|
},
|
2020-03-26 21:07:15 +00:00
|
|
|
SignerName: certificates.KubeAPIServerClientKubeletSignerName,
|
2019-01-12 04:58:27 +00:00
|
|
|
Usages: []certificates.KeyUsage{
|
|
|
|
// https://tools.ietf.org/html/rfc5280#section-4.2.1.3
|
|
|
|
//
|
|
|
|
// DigitalSignature allows the certificate to be used to verify
|
|
|
|
// digital signatures including signatures used during TLS
|
|
|
|
// negotiation.
|
|
|
|
certificates.UsageDigitalSignature,
|
|
|
|
// KeyEncipherment allows the cert/key pair to be used to encrypt
|
|
|
|
// keys, including the symmetric keys negotiated during TLS setup
|
|
|
|
// and used for data transfer..
|
|
|
|
certificates.UsageKeyEncipherment,
|
|
|
|
// ClientAuth allows the cert to be used by a TLS client to
|
|
|
|
// authenticate itself to the TLS server.
|
|
|
|
certificates.UsageClientAuth,
|
|
|
|
},
|
2019-04-07 17:07:55 +00:00
|
|
|
|
|
|
|
// For backwards compatibility, the kubelet supports the ability to
|
|
|
|
// provide a higher privileged certificate as initial data that will
|
|
|
|
// then be rotated immediately. This code path is used by kubeadm on
|
|
|
|
// the masters.
|
|
|
|
BootstrapCertificatePEM: bootstrapCertData,
|
|
|
|
BootstrapKeyPEM: bootstrapKeyData,
|
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
CertificateStore: certificateStore,
|
|
|
|
CertificateRenewFailure: certificateRenewFailure,
|
2019-01-12 04:58:27 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return nil, fmt.Errorf("failed to initialize client certificate manager: %v", err)
|
|
|
|
}
|
2020-08-10 17:43:49 +00:00
|
|
|
|
2019-01-12 04:58:27 +00:00
|
|
|
return m, nil
|
|
|
|
}
|