2019-01-12 04:58:27 +00:00
|
|
|
/*
|
|
|
|
Copyright 2016 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 approver implements an automated approver for kubelet certificates.
|
|
|
|
package approver
|
|
|
|
|
|
|
|
import (
|
2020-03-26 21:07:15 +00:00
|
|
|
"context"
|
2019-01-12 04:58:27 +00:00
|
|
|
"crypto/x509"
|
|
|
|
"fmt"
|
|
|
|
|
2020-03-26 21:07:15 +00:00
|
|
|
authorization "k8s.io/api/authorization/v1"
|
2020-08-10 17:43:49 +00:00
|
|
|
capi "k8s.io/api/certificates/v1"
|
|
|
|
corev1 "k8s.io/api/core/v1"
|
2020-03-26 21:07:15 +00:00
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
2020-08-10 17:43:49 +00:00
|
|
|
"k8s.io/apimachinery/pkg/util/sets"
|
|
|
|
certificatesinformers "k8s.io/client-go/informers/certificates/v1"
|
2019-01-12 04:58:27 +00:00
|
|
|
clientset "k8s.io/client-go/kubernetes"
|
2020-03-26 21:07:15 +00:00
|
|
|
|
2020-08-10 17:43:49 +00:00
|
|
|
capihelper "k8s.io/kubernetes/pkg/apis/certificates"
|
2019-01-12 04:58:27 +00:00
|
|
|
"k8s.io/kubernetes/pkg/controller/certificates"
|
|
|
|
)
|
|
|
|
|
|
|
|
type csrRecognizer struct {
|
|
|
|
recognize func(csr *capi.CertificateSigningRequest, x509cr *x509.CertificateRequest) bool
|
|
|
|
permission authorization.ResourceAttributes
|
|
|
|
successMessage string
|
|
|
|
}
|
|
|
|
|
|
|
|
type sarApprover struct {
|
|
|
|
client clientset.Interface
|
|
|
|
recognizers []csrRecognizer
|
|
|
|
}
|
|
|
|
|
2019-08-30 18:33:25 +00:00
|
|
|
// NewCSRApprovingController creates a new CSRApprovingController.
|
2019-01-12 04:58:27 +00:00
|
|
|
func NewCSRApprovingController(client clientset.Interface, csrInformer certificatesinformers.CertificateSigningRequestInformer) *certificates.CertificateController {
|
|
|
|
approver := &sarApprover{
|
|
|
|
client: client,
|
|
|
|
recognizers: recognizers(),
|
|
|
|
}
|
|
|
|
return certificates.NewCertificateController(
|
2019-12-12 01:27:03 +00:00
|
|
|
"csrapproving",
|
2019-01-12 04:58:27 +00:00
|
|
|
client,
|
|
|
|
csrInformer,
|
|
|
|
approver.handle,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
|
|
|
func recognizers() []csrRecognizer {
|
|
|
|
recognizers := []csrRecognizer{
|
|
|
|
{
|
|
|
|
recognize: isSelfNodeClientCert,
|
|
|
|
permission: authorization.ResourceAttributes{Group: "certificates.k8s.io", Resource: "certificatesigningrequests", Verb: "create", Subresource: "selfnodeclient"},
|
|
|
|
successMessage: "Auto approving self kubelet client certificate after SubjectAccessReview.",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
recognize: isNodeClientCert,
|
|
|
|
permission: authorization.ResourceAttributes{Group: "certificates.k8s.io", Resource: "certificatesigningrequests", Verb: "create", Subresource: "nodeclient"},
|
|
|
|
successMessage: "Auto approving kubelet client certificate after SubjectAccessReview.",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
return recognizers
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *sarApprover) handle(csr *capi.CertificateSigningRequest) error {
|
|
|
|
if len(csr.Status.Certificate) != 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
if approved, denied := certificates.GetCertApprovalCondition(&csr.Status); approved || denied {
|
|
|
|
return nil
|
|
|
|
}
|
2020-03-26 21:07:15 +00:00
|
|
|
x509cr, err := capihelper.ParseCSR(csr.Spec.Request)
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("unable to parse csr %q: %v", csr.Name, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
tried := []string{}
|
|
|
|
|
|
|
|
for _, r := range a.recognizers {
|
|
|
|
if !r.recognize(csr, x509cr) {
|
|
|
|
continue
|
|
|
|
}
|
|
|
|
|
|
|
|
tried = append(tried, r.permission.Subresource)
|
|
|
|
|
|
|
|
approved, err := a.authorize(csr, r.permission)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
if approved {
|
|
|
|
appendApprovalCondition(csr, r.successMessage)
|
2020-08-10 17:43:49 +00:00
|
|
|
_, err = a.client.CertificatesV1().CertificateSigningRequests().UpdateApproval(context.Background(), csr.Name, csr, metav1.UpdateOptions{})
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return fmt.Errorf("error updating approval for csr: %v", err)
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if len(tried) != 0 {
|
|
|
|
return certificates.IgnorableError("recognized csr %q as %v but subject access review was not approved", csr.Name, tried)
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (a *sarApprover) authorize(csr *capi.CertificateSigningRequest, rattrs authorization.ResourceAttributes) (bool, error) {
|
|
|
|
extra := make(map[string]authorization.ExtraValue)
|
|
|
|
for k, v := range csr.Spec.Extra {
|
|
|
|
extra[k] = authorization.ExtraValue(v)
|
|
|
|
}
|
|
|
|
|
|
|
|
sar := &authorization.SubjectAccessReview{
|
|
|
|
Spec: authorization.SubjectAccessReviewSpec{
|
|
|
|
User: csr.Spec.Username,
|
|
|
|
UID: csr.Spec.UID,
|
|
|
|
Groups: csr.Spec.Groups,
|
|
|
|
Extra: extra,
|
|
|
|
ResourceAttributes: &rattrs,
|
|
|
|
},
|
|
|
|
}
|
2020-03-26 21:07:15 +00:00
|
|
|
sar, err := a.client.AuthorizationV1().SubjectAccessReviews().Create(context.TODO(), sar, metav1.CreateOptions{})
|
2019-01-12 04:58:27 +00:00
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
return sar.Status.Allowed, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func appendApprovalCondition(csr *capi.CertificateSigningRequest, message string) {
|
|
|
|
csr.Status.Conditions = append(csr.Status.Conditions, capi.CertificateSigningRequestCondition{
|
|
|
|
Type: capi.CertificateApproved,
|
2020-08-10 17:43:49 +00:00
|
|
|
Status: corev1.ConditionTrue,
|
2019-01-12 04:58:27 +00:00
|
|
|
Reason: "AutoApproved",
|
|
|
|
Message: message,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
func isNodeClientCert(csr *capi.CertificateSigningRequest, x509cr *x509.CertificateRequest) bool {
|
2020-08-10 17:43:49 +00:00
|
|
|
if csr.Spec.SignerName != capi.KubeAPIServerClientKubeletSignerName {
|
2019-01-12 04:58:27 +00:00
|
|
|
return false
|
|
|
|
}
|
2020-08-10 17:43:49 +00:00
|
|
|
return capihelper.IsKubeletClientCSR(x509cr, usagesToSet(csr.Spec.Usages))
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
func isSelfNodeClientCert(csr *capi.CertificateSigningRequest, x509cr *x509.CertificateRequest) bool {
|
|
|
|
if csr.Spec.Username != x509cr.Subject.CommonName {
|
|
|
|
return false
|
|
|
|
}
|
2020-08-10 17:43:49 +00:00
|
|
|
return isNodeClientCert(csr, x509cr)
|
|
|
|
}
|
|
|
|
|
|
|
|
func usagesToSet(usages []capi.KeyUsage) sets.String {
|
|
|
|
result := sets.NewString()
|
|
|
|
for _, usage := range usages {
|
|
|
|
result.Insert(string(usage))
|
|
|
|
}
|
|
|
|
return result
|
2019-01-12 04:58:27 +00:00
|
|
|
}
|