mirror of
				https://github.com/k3s-io/kubernetes.git
				synced 2025-10-26 11:07:45 +00:00 
			
		
		
		
	kube-apiserver: add a bootstrap token authenticator for TLS bootstrapping
This commit is contained in:
		
							
								
								
									
										168
									
								
								plugin/pkg/auth/authenticator/token/bootstrap/bootstrap.go
									
									
									
									
									
										Normal file
									
								
							
							
						
						
									
										168
									
								
								plugin/pkg/auth/authenticator/token/bootstrap/bootstrap.go
									
									
									
									
									
										Normal file
									
								
							| @@ -0,0 +1,168 @@ | ||||
| /* | ||||
| 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 bootstrap provides a token authenticator for TLS bootstrap secrets. | ||||
| */ | ||||
| package bootstrap | ||||
|  | ||||
| import ( | ||||
| 	"fmt" | ||||
| 	"regexp" | ||||
| 	"time" | ||||
|  | ||||
| 	"github.com/golang/glog" | ||||
|  | ||||
| 	"k8s.io/apimachinery/pkg/api/errors" | ||||
| 	"k8s.io/apiserver/pkg/authentication/user" | ||||
| 	"k8s.io/kubernetes/pkg/api" | ||||
| 	bootstrapapi "k8s.io/kubernetes/pkg/bootstrap/api" | ||||
| 	"k8s.io/kubernetes/pkg/client/listers/core/internalversion" | ||||
| ) | ||||
|  | ||||
| // TODO: A few methods in this package is copied from other sources. Either | ||||
| // because the existing functionality isn't exported or because it is in a | ||||
| // package that shouldn't be directly imported by this packages. | ||||
|  | ||||
| // NewTokenAuthenticator initializes a bootstrap token authenticator. | ||||
| // | ||||
| // Lister is expected to be for the "kube-system" namespace. | ||||
| func NewTokenAuthenticator(lister internalversion.SecretNamespaceLister) *TokenAuthenticator { | ||||
| 	return &TokenAuthenticator{lister} | ||||
| } | ||||
|  | ||||
| // TokenAuthenticator authenticates bootstrap tokens from secrets in the API server. | ||||
| type TokenAuthenticator struct { | ||||
| 	lister internalversion.SecretNamespaceLister | ||||
| } | ||||
|  | ||||
| // AuthenticateToken tries to match the provided token to a bootstrap token secret | ||||
| // in a given namespace. If found, it authenticates the token in the | ||||
| // "system:bootstrappers" group and with the "system:bootstrap:(token-id)" username. | ||||
| // | ||||
| // All secrets must be of type "bootstrap.kubernetes.io/token". An example secret: | ||||
| // | ||||
| //     apiVersion: v1 | ||||
| //     kind: Secret | ||||
| //     metadata: | ||||
| //       # Name MUST be of form "bootstrap-token-( token id )". | ||||
| //       name: bootstrap-token-( token id ) | ||||
| //       namespace: kube-system | ||||
| //     # Only secrets of this type will be evaluated. | ||||
| //     type: bootstrap.kubernetes.io/token | ||||
| //     data: | ||||
| //       token-secret: ( private part of token ) | ||||
| //       token-id: ( token id ) | ||||
| //       # Required key usage. | ||||
| //       usage-bootstrap-authentication: true | ||||
| //       # May also contain an expiry. | ||||
| // | ||||
| // Tokens are expected to be of the form: | ||||
| // | ||||
| //     ( token-id ).( token-secret ) | ||||
| // | ||||
| func (t *TokenAuthenticator) AuthenticateToken(token string) (user.Info, bool, error) { | ||||
| 	tokenID, tokenSecret, err := parseToken(token) | ||||
| 	if err != nil { | ||||
| 		// Token isn't of the correct form, ignore it. | ||||
| 		return nil, false, nil | ||||
| 	} | ||||
|  | ||||
| 	secretName := bootstrapapi.BootstrapTokenSecretPrefix + tokenID | ||||
| 	secret, err := t.lister.Get(secretName) | ||||
| 	if err != nil { | ||||
| 		if errors.IsNotFound(err) { | ||||
| 			return nil, false, nil | ||||
| 		} | ||||
| 		return nil, false, err | ||||
| 	} | ||||
|  | ||||
| 	if string(secret.Type) != string(bootstrapapi.SecretTypeBootstrapToken) || secret.Data == nil { | ||||
| 		return nil, false, nil | ||||
| 	} | ||||
|  | ||||
| 	ts := getSecretString(secret, bootstrapapi.BootstrapTokenSecretKey) | ||||
| 	if ts != tokenSecret { | ||||
| 		return nil, false, nil | ||||
| 	} | ||||
|  | ||||
| 	id := getSecretString(secret, bootstrapapi.BootstrapTokenIDKey) | ||||
| 	if id != tokenID { | ||||
| 		return nil, false, nil | ||||
| 	} | ||||
|  | ||||
| 	if isSecretExpired(secret) { | ||||
| 		return nil, false, nil | ||||
| 	} | ||||
|  | ||||
| 	if getSecretString(secret, bootstrapapi.BootstrapTokenUsageAuthentication) != "true" { | ||||
| 		glog.V(3).Infof("Bearer token matching bootstrap Secret %s/%s not marked %s=true.", | ||||
| 			secret.Namespace, secret.Name, bootstrapapi.BootstrapTokenUsageAuthentication) | ||||
| 		return nil, false, nil | ||||
| 	} | ||||
|  | ||||
| 	return &user.DefaultInfo{ | ||||
| 		Name:   bootstrapapi.BootstrapUserPrefix + string(id), | ||||
| 		Groups: []string{bootstrapapi.BootstrapGroup}, | ||||
| 	}, true, nil | ||||
| } | ||||
|  | ||||
| // Copied from k8s.io/kubernetes/pkg/bootstrap/api | ||||
| func getSecretString(secret *api.Secret, key string) string { | ||||
| 	if secret.Data == nil { | ||||
| 		return "" | ||||
| 	} | ||||
| 	if val, ok := secret.Data[key]; ok { | ||||
| 		return string(val) | ||||
| 	} | ||||
| 	return "" | ||||
| } | ||||
|  | ||||
| // Copied from k8s.io/kubernetes/pkg/bootstrap/api | ||||
| func isSecretExpired(secret *api.Secret) bool { | ||||
| 	expiration := getSecretString(secret, bootstrapapi.BootstrapTokenExpirationKey) | ||||
| 	if len(expiration) > 0 { | ||||
| 		expTime, err2 := time.Parse(time.RFC3339, expiration) | ||||
| 		if err2 != nil { | ||||
| 			glog.V(3).Infof("Unparseable expiration time (%s) in %s/%s Secret: %v. Treating as expired.", | ||||
| 				expiration, secret.Namespace, secret.Name, err2) | ||||
| 			return true | ||||
| 		} | ||||
| 		if time.Now().After(expTime) { | ||||
| 			glog.V(3).Infof("Expired bootstrap token in %s/%s Secret: %v", | ||||
| 				secret.Namespace, secret.Name, expiration) | ||||
| 			return true | ||||
| 		} | ||||
| 	} | ||||
| 	return false | ||||
| } | ||||
|  | ||||
| // Copied from kubernetes/cmd/kubeadm/app/util/token | ||||
|  | ||||
| var ( | ||||
| 	tokenRegexpString = "^([a-z0-9]{6})\\.([a-z0-9]{16})$" | ||||
| 	tokenRegexp       = regexp.MustCompile(tokenRegexpString) | ||||
| ) | ||||
|  | ||||
| // parseToken tries and parse a valid token from a string. | ||||
| // A token ID and token secret are returned in case of success, an error otherwise. | ||||
| func parseToken(s string) (string, string, error) { | ||||
| 	split := tokenRegexp.FindStringSubmatch(s) | ||||
| 	if len(split) != 3 { | ||||
| 		return "", "", fmt.Errorf("token [%q] was not of form [%q]", s, tokenRegexpString) | ||||
| 	} | ||||
| 	return split[1], split[2], nil | ||||
| } | ||||
		Reference in New Issue
	
	Block a user