2023-04-03 23:22:21 +00:00
|
|
|
// Copyright 2020-2023 the Pinniped contributors. All Rights Reserved.
|
2021-02-19 18:21:10 +00:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
package integration
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
|
|
|
"crypto/ecdsa"
|
|
|
|
"crypto/elliptic"
|
|
|
|
"crypto/rand"
|
|
|
|
"crypto/x509"
|
|
|
|
"crypto/x509/pkix"
|
|
|
|
"encoding/pem"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
authenticationv1 "k8s.io/api/authentication/v1"
|
|
|
|
certificatesv1 "k8s.io/api/certificates/v1"
|
2021-02-23 04:30:02 +00:00
|
|
|
certificatesv1beta1 "k8s.io/api/certificates/v1beta1"
|
2021-02-19 18:21:10 +00:00
|
|
|
corev1 "k8s.io/api/core/v1"
|
|
|
|
"k8s.io/apimachinery/pkg/api/errors"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
|
|
|
"k8s.io/client-go/rest"
|
|
|
|
"k8s.io/client-go/util/cert"
|
|
|
|
"k8s.io/client-go/util/certificate/csr"
|
|
|
|
"k8s.io/client-go/util/keyutil"
|
|
|
|
|
|
|
|
identityv1alpha1 "go.pinniped.dev/generated/latest/apis/concierge/identity/v1alpha1"
|
2021-09-21 00:14:58 +00:00
|
|
|
"go.pinniped.dev/internal/testutil"
|
2021-06-22 15:23:19 +00:00
|
|
|
"go.pinniped.dev/test/testlib"
|
2021-02-19 18:21:10 +00:00
|
|
|
)
|
|
|
|
|
2021-08-25 19:34:33 +00:00
|
|
|
// whoami requests are non-mutating and safe to run in parallel with serial tests, see main_test.go.
|
|
|
|
func TestWhoAmI_Kubeadm_Parallel(t *testing.T) {
|
2021-02-19 18:21:10 +00:00
|
|
|
// use the cluster signing key being available as a proxy for this being a kubeadm cluster
|
|
|
|
// we should add more robust logic around skipping clusters based on vendor
|
2021-06-22 15:23:19 +00:00
|
|
|
_ = testlib.IntegrationEnv(t).WithCapability(testlib.ClusterSigningKeyIsAvailable)
|
2021-02-19 18:21:10 +00:00
|
|
|
|
2021-03-05 01:25:43 +00:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), time.Minute)
|
2021-02-19 18:21:10 +00:00
|
|
|
defer cancel()
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
whoAmI, err := testlib.NewConciergeClientset(t).IdentityV1alpha1().WhoAmIRequests().
|
2021-02-19 18:21:10 +00:00
|
|
|
Create(ctx, &identityv1alpha1.WhoAmIRequest{}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// this user info is based off of the bootstrap cert user created by kubeadm
|
|
|
|
require.Equal(t,
|
|
|
|
&identityv1alpha1.WhoAmIRequest{
|
|
|
|
Status: identityv1alpha1.WhoAmIRequestStatus{
|
|
|
|
KubernetesUserInfo: identityv1alpha1.KubernetesUserInfo{
|
|
|
|
User: identityv1alpha1.UserInfo{
|
|
|
|
Username: "kubernetes-admin",
|
|
|
|
Groups: []string{
|
|
|
|
"system:masters",
|
|
|
|
"system:authenticated",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
whoAmI,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-08-25 19:34:33 +00:00
|
|
|
// whoami requests are non-mutating and safe to run in parallel with serial tests, see main_test.go.
|
|
|
|
func TestWhoAmI_ServiceAccount_Legacy_Parallel(t *testing.T) {
|
2021-06-22 15:23:19 +00:00
|
|
|
_ = testlib.IntegrationEnv(t)
|
2021-02-19 18:21:10 +00:00
|
|
|
|
2021-03-05 01:25:43 +00:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), time.Minute)
|
2021-02-19 18:21:10 +00:00
|
|
|
defer cancel()
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
kubeClient := testlib.NewKubernetesClientset(t).CoreV1()
|
2021-02-19 18:21:10 +00:00
|
|
|
|
|
|
|
ns, err := kubeClient.Namespaces().Create(ctx, &corev1.Namespace{
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
GenerateName: "test-whoami-",
|
|
|
|
},
|
|
|
|
}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-03-11 18:01:17 +00:00
|
|
|
t.Cleanup(func() {
|
|
|
|
require.NoError(t, kubeClient.Namespaces().Delete(context.Background(), ns.Name, metav1.DeleteOptions{}))
|
|
|
|
})
|
2021-02-19 18:21:10 +00:00
|
|
|
|
|
|
|
sa, err := kubeClient.ServiceAccounts(ns.Name).Create(ctx, &corev1.ServiceAccount{
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
GenerateName: "test-whoami-",
|
|
|
|
},
|
|
|
|
}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2023-10-20 20:28:53 +00:00
|
|
|
// TODO: What is this used for?
|
2021-02-19 18:21:10 +00:00
|
|
|
secret, err := kubeClient.Secrets(ns.Name).Create(ctx, &corev1.Secret{
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
GenerateName: "test-whoami-",
|
|
|
|
Annotations: map[string]string{
|
|
|
|
corev1.ServiceAccountNameKey: sa.Name,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
Type: corev1.SecretTypeServiceAccountToken,
|
|
|
|
}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
testlib.RequireEventuallyWithoutError(t, func() (bool, error) {
|
2021-02-19 18:21:10 +00:00
|
|
|
secret, err = kubeClient.Secrets(ns.Name).Get(ctx, secret.Name, metav1.GetOptions{})
|
|
|
|
if err != nil {
|
|
|
|
return false, err
|
|
|
|
}
|
|
|
|
return len(secret.Data[corev1.ServiceAccountTokenKey]) > 0, nil
|
2021-03-05 01:25:43 +00:00
|
|
|
}, time.Minute, time.Second)
|
2021-02-19 18:21:10 +00:00
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
saConfig := testlib.NewAnonymousClientRestConfig(t)
|
2021-02-19 18:21:10 +00:00
|
|
|
saConfig.BearerToken = string(secret.Data[corev1.ServiceAccountTokenKey])
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
whoAmI, err := testlib.NewKubeclient(t, saConfig).PinnipedConcierge.IdentityV1alpha1().WhoAmIRequests().
|
2021-02-19 18:21:10 +00:00
|
|
|
Create(ctx, &identityv1alpha1.WhoAmIRequest{}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// legacy service account tokens do not have any extra info
|
|
|
|
require.Equal(t,
|
|
|
|
&identityv1alpha1.WhoAmIRequest{
|
|
|
|
Status: identityv1alpha1.WhoAmIRequestStatus{
|
|
|
|
KubernetesUserInfo: identityv1alpha1.KubernetesUserInfo{
|
|
|
|
User: identityv1alpha1.UserInfo{
|
|
|
|
Username: "system:serviceaccount:" + ns.Name + ":" + sa.Name,
|
|
|
|
UID: "", // aggregation drops UID: https://github.com/kubernetes/kubernetes/issues/93699
|
|
|
|
Groups: []string{
|
|
|
|
"system:serviceaccounts",
|
|
|
|
"system:serviceaccounts:" + ns.Name,
|
|
|
|
"system:authenticated",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
whoAmI,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-08-25 19:34:33 +00:00
|
|
|
// whoami requests are non-mutating and safe to run in parallel with serial tests, see main_test.go.
|
|
|
|
func TestWhoAmI_ServiceAccount_TokenRequest_Parallel(t *testing.T) {
|
2021-08-25 22:12:07 +00:00
|
|
|
env := testlib.IntegrationEnv(t)
|
2021-02-19 18:21:10 +00:00
|
|
|
|
2021-03-05 01:25:43 +00:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), time.Minute)
|
2021-02-19 18:21:10 +00:00
|
|
|
defer cancel()
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
kubeClient := testlib.NewKubernetesClientset(t).CoreV1()
|
2021-02-19 18:21:10 +00:00
|
|
|
|
|
|
|
ns, err := kubeClient.Namespaces().Create(ctx, &corev1.Namespace{
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
GenerateName: "test-whoami-",
|
|
|
|
},
|
|
|
|
}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-03-11 18:01:17 +00:00
|
|
|
t.Cleanup(func() {
|
|
|
|
require.NoError(t, kubeClient.Namespaces().Delete(context.Background(), ns.Name, metav1.DeleteOptions{}))
|
|
|
|
})
|
2021-02-19 18:21:10 +00:00
|
|
|
|
|
|
|
sa, err := kubeClient.ServiceAccounts(ns.Name).Create(ctx, &corev1.ServiceAccount{
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
GenerateName: "test-whoami-",
|
|
|
|
},
|
|
|
|
}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
_, tokenRequestProbeErr := kubeClient.ServiceAccounts(ns.Name).CreateToken(ctx, sa.Name, &authenticationv1.TokenRequest{}, metav1.CreateOptions{})
|
|
|
|
if errors.IsNotFound(tokenRequestProbeErr) && tokenRequestProbeErr.Error() == "the server could not find the requested resource" {
|
|
|
|
return // stop test early since the token request API is not enabled on this cluster - other errors are caught below
|
|
|
|
}
|
|
|
|
|
2021-08-17 00:47:08 +00:00
|
|
|
pod := testlib.CreatePod(ctx, t, "whoami", ns.Name,
|
|
|
|
corev1.PodSpec{
|
2021-02-19 18:21:10 +00:00
|
|
|
Containers: []corev1.Container{
|
|
|
|
{
|
2021-08-25 22:12:07 +00:00
|
|
|
Name: "sleeper",
|
|
|
|
Image: env.ShellContainerImage,
|
|
|
|
ImagePullPolicy: corev1.PullIfNotPresent,
|
|
|
|
Command: []string{"sh", "-c", "sleep 3600"},
|
2022-09-15 21:58:15 +00:00
|
|
|
// Use a restrictive security context just in case the test cluster has PSAs enabled.
|
|
|
|
SecurityContext: testlib.RestrictiveSecurityContext(),
|
2021-02-19 18:21:10 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
ServiceAccountName: sa.Name,
|
2021-08-17 00:47:08 +00:00
|
|
|
})
|
2021-02-19 18:21:10 +00:00
|
|
|
|
|
|
|
tokenRequestBadAudience, err := kubeClient.ServiceAccounts(ns.Name).CreateToken(ctx, sa.Name, &authenticationv1.TokenRequest{
|
|
|
|
Spec: authenticationv1.TokenRequestSpec{
|
|
|
|
Audiences: []string{"should-fail-because-wrong-audience"}, // anything that is not an API server audience
|
|
|
|
BoundObjectRef: &authenticationv1.BoundObjectReference{
|
|
|
|
Kind: "Pod",
|
|
|
|
APIVersion: "",
|
|
|
|
Name: pod.Name,
|
|
|
|
UID: pod.UID,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
saBadAudConfig := testlib.NewAnonymousClientRestConfig(t)
|
2021-02-19 18:21:10 +00:00
|
|
|
saBadAudConfig.BearerToken = tokenRequestBadAudience.Status.Token
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
_, badAudErr := testlib.NewKubeclient(t, saBadAudConfig).PinnipedConcierge.IdentityV1alpha1().WhoAmIRequests().
|
2021-02-19 18:21:10 +00:00
|
|
|
Create(ctx, &identityv1alpha1.WhoAmIRequest{}, metav1.CreateOptions{})
|
2021-06-22 15:23:19 +00:00
|
|
|
require.True(t, errors.IsUnauthorized(badAudErr), testlib.Sdump(badAudErr))
|
2021-02-19 18:21:10 +00:00
|
|
|
|
|
|
|
tokenRequest, err := kubeClient.ServiceAccounts(ns.Name).CreateToken(ctx, sa.Name, &authenticationv1.TokenRequest{
|
|
|
|
Spec: authenticationv1.TokenRequestSpec{
|
|
|
|
Audiences: []string{},
|
|
|
|
BoundObjectRef: &authenticationv1.BoundObjectReference{
|
|
|
|
Kind: "Pod",
|
|
|
|
APIVersion: "",
|
|
|
|
Name: pod.Name,
|
|
|
|
UID: pod.UID,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
saTokenReqConfig := testlib.NewAnonymousClientRestConfig(t)
|
2021-02-19 18:21:10 +00:00
|
|
|
saTokenReqConfig.BearerToken = tokenRequest.Status.Token
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
whoAmITokenReq, err := testlib.NewKubeclient(t, saTokenReqConfig).PinnipedConcierge.IdentityV1alpha1().WhoAmIRequests().
|
2021-02-19 18:21:10 +00:00
|
|
|
Create(ctx, &identityv1alpha1.WhoAmIRequest{}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// new service account tokens include the pod info in the extra fields
|
|
|
|
require.Equal(t,
|
|
|
|
&identityv1alpha1.WhoAmIRequest{
|
|
|
|
Status: identityv1alpha1.WhoAmIRequestStatus{
|
|
|
|
KubernetesUserInfo: identityv1alpha1.KubernetesUserInfo{
|
|
|
|
User: identityv1alpha1.UserInfo{
|
|
|
|
Username: "system:serviceaccount:" + ns.Name + ":" + sa.Name,
|
|
|
|
UID: "", // aggregation drops UID: https://github.com/kubernetes/kubernetes/issues/93699
|
|
|
|
Groups: []string{
|
|
|
|
"system:serviceaccounts",
|
|
|
|
"system:serviceaccounts:" + ns.Name,
|
|
|
|
"system:authenticated",
|
|
|
|
},
|
|
|
|
Extra: map[string]identityv1alpha1.ExtraValue{
|
|
|
|
"authentication.kubernetes.io/pod-name": {pod.Name},
|
|
|
|
"authentication.kubernetes.io/pod-uid": {string(pod.UID)},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
whoAmITokenReq,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-08-25 19:34:33 +00:00
|
|
|
// whoami requests are non-mutating and safe to run in parallel with serial tests, see main_test.go.
|
|
|
|
func TestWhoAmI_CSR_Parallel(t *testing.T) {
|
2021-02-19 18:21:10 +00:00
|
|
|
// use the cluster signing key being available as a proxy for this not being an EKS cluster
|
|
|
|
// we should add more robust logic around skipping clusters based on vendor
|
2021-06-22 15:23:19 +00:00
|
|
|
_ = testlib.IntegrationEnv(t).WithCapability(testlib.ClusterSigningKeyIsAvailable)
|
2021-02-19 18:21:10 +00:00
|
|
|
|
2021-03-05 01:25:43 +00:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), time.Minute)
|
2021-02-19 18:21:10 +00:00
|
|
|
defer cancel()
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
kubeClient := testlib.NewKubernetesClientset(t)
|
2021-02-19 18:21:10 +00:00
|
|
|
|
|
|
|
privateKey, err := ecdsa.GenerateKey(elliptic.P256(), rand.Reader)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
der, err := x509.MarshalECPrivateKey(privateKey)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
keyPEM := pem.EncodeToMemory(&pem.Block{Type: keyutil.ECPrivateKeyBlockType, Bytes: der})
|
|
|
|
|
|
|
|
csrPEM, err := cert.MakeCSR(privateKey, &pkix.Name{
|
|
|
|
CommonName: "panda-man",
|
|
|
|
Organization: []string{"living-the-dream", "need-more-sleep"},
|
|
|
|
}, nil, nil)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
csrName, csrUID, err := csr.RequestCertificate(
|
|
|
|
kubeClient,
|
|
|
|
csrPEM,
|
|
|
|
"",
|
|
|
|
certificatesv1.KubeAPIServerClientSignerName,
|
2021-08-09 23:16:50 +00:00
|
|
|
nil,
|
2021-02-19 18:21:10 +00:00
|
|
|
[]certificatesv1.KeyUsage{certificatesv1.UsageClientAuth},
|
|
|
|
privateKey,
|
|
|
|
)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-09-21 00:14:58 +00:00
|
|
|
useCertificatesV1API := testutil.KubeServerSupportsCertificatesV1API(t, kubeClient.Discovery())
|
|
|
|
|
2021-03-11 18:01:17 +00:00
|
|
|
t.Cleanup(func() {
|
2021-09-21 00:14:58 +00:00
|
|
|
if useCertificatesV1API {
|
|
|
|
require.NoError(t, kubeClient.CertificatesV1().CertificateSigningRequests().
|
|
|
|
Delete(context.Background(), csrName, metav1.DeleteOptions{}))
|
|
|
|
} else {
|
|
|
|
// On old clusters use v1beta1
|
|
|
|
require.NoError(t, kubeClient.CertificatesV1beta1().CertificateSigningRequests().
|
|
|
|
Delete(context.Background(), csrName, metav1.DeleteOptions{}))
|
|
|
|
}
|
2021-03-11 18:01:17 +00:00
|
|
|
})
|
2021-02-19 18:21:10 +00:00
|
|
|
|
2021-09-21 00:14:58 +00:00
|
|
|
if useCertificatesV1API {
|
|
|
|
_, err = kubeClient.CertificatesV1().CertificateSigningRequests().UpdateApproval(ctx, csrName, &certificatesv1.CertificateSigningRequest{
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
Name: csrName,
|
|
|
|
},
|
|
|
|
Status: certificatesv1.CertificateSigningRequestStatus{
|
|
|
|
Conditions: []certificatesv1.CertificateSigningRequestCondition{
|
|
|
|
{
|
|
|
|
Type: certificatesv1.CertificateApproved,
|
|
|
|
Status: corev1.ConditionTrue,
|
|
|
|
Reason: "WhoAmICSRTest",
|
|
|
|
},
|
2021-02-19 18:21:10 +00:00
|
|
|
},
|
|
|
|
},
|
2021-09-21 00:14:58 +00:00
|
|
|
}, metav1.UpdateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
} else {
|
|
|
|
// On old Kubernetes clusters use CertificatesV1beta1
|
|
|
|
_, err = kubeClient.CertificatesV1beta1().CertificateSigningRequests().UpdateApproval(ctx, &certificatesv1beta1.CertificateSigningRequest{
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
Name: csrName,
|
|
|
|
},
|
|
|
|
Status: certificatesv1beta1.CertificateSigningRequestStatus{
|
|
|
|
Conditions: []certificatesv1beta1.CertificateSigningRequestCondition{
|
|
|
|
{
|
|
|
|
Type: certificatesv1beta1.CertificateApproved,
|
|
|
|
Status: corev1.ConditionTrue,
|
|
|
|
Reason: "WhoAmICSRTest",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}, metav1.UpdateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
2021-02-19 18:21:10 +00:00
|
|
|
|
|
|
|
crtPEM, err := csr.WaitForCertificate(ctx, kubeClient, csrName, csrUID)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
csrConfig := testlib.NewAnonymousClientRestConfig(t)
|
2021-02-19 18:21:10 +00:00
|
|
|
csrConfig.CertData = crtPEM
|
|
|
|
csrConfig.KeyData = keyPEM
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
whoAmI, err := testlib.NewKubeclient(t, csrConfig).PinnipedConcierge.IdentityV1alpha1().WhoAmIRequests().
|
2021-02-19 18:21:10 +00:00
|
|
|
Create(ctx, &identityv1alpha1.WhoAmIRequest{}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
require.Equal(t,
|
|
|
|
&identityv1alpha1.WhoAmIRequest{
|
|
|
|
Status: identityv1alpha1.WhoAmIRequestStatus{
|
|
|
|
KubernetesUserInfo: identityv1alpha1.KubernetesUserInfo{
|
|
|
|
User: identityv1alpha1.UserInfo{
|
|
|
|
Username: "panda-man",
|
|
|
|
Groups: []string{
|
|
|
|
"need-more-sleep",
|
|
|
|
"living-the-dream",
|
|
|
|
"system:authenticated",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
whoAmI,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-08-25 19:34:33 +00:00
|
|
|
// whoami requests are non-mutating and safe to run in parallel with serial tests, see main_test.go.
|
|
|
|
func TestWhoAmI_Anonymous_Parallel(t *testing.T) {
|
2021-06-22 15:23:19 +00:00
|
|
|
_ = testlib.IntegrationEnv(t).WithCapability(testlib.AnonymousAuthenticationSupported)
|
2021-02-19 18:21:10 +00:00
|
|
|
|
2021-03-05 01:25:43 +00:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), time.Minute)
|
2021-02-19 18:21:10 +00:00
|
|
|
defer cancel()
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
anonymousConfig := testlib.NewAnonymousClientRestConfig(t)
|
2021-02-19 18:21:10 +00:00
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
whoAmI, err := testlib.NewKubeclient(t, anonymousConfig).PinnipedConcierge.IdentityV1alpha1().WhoAmIRequests().
|
2021-02-19 18:21:10 +00:00
|
|
|
Create(ctx, &identityv1alpha1.WhoAmIRequest{}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// this also asserts that all users, even unauthenticated ones, can call this API when anonymous is enabled
|
|
|
|
require.Equal(t,
|
|
|
|
&identityv1alpha1.WhoAmIRequest{
|
|
|
|
Status: identityv1alpha1.WhoAmIRequestStatus{
|
|
|
|
KubernetesUserInfo: identityv1alpha1.KubernetesUserInfo{
|
|
|
|
User: identityv1alpha1.UserInfo{
|
|
|
|
Username: "system:anonymous",
|
|
|
|
Groups: []string{
|
|
|
|
"system:unauthenticated",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
whoAmI,
|
|
|
|
)
|
|
|
|
}
|
|
|
|
|
2021-08-25 19:34:33 +00:00
|
|
|
// whoami requests are non-mutating and safe to run in parallel with serial tests, see main_test.go.
|
|
|
|
func TestWhoAmI_ImpersonateDirectly_Parallel(t *testing.T) {
|
2021-06-22 15:23:19 +00:00
|
|
|
_ = testlib.IntegrationEnv(t)
|
2021-02-19 18:21:10 +00:00
|
|
|
|
2021-03-05 01:25:43 +00:00
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), time.Minute)
|
2021-02-19 18:21:10 +00:00
|
|
|
defer cancel()
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
impersonationConfig := testlib.NewClientConfig(t)
|
2021-02-19 18:21:10 +00:00
|
|
|
impersonationConfig.Impersonate = rest.ImpersonationConfig{
|
|
|
|
UserName: "solaire",
|
2021-02-23 04:30:02 +00:00
|
|
|
// need to impersonate system:authenticated directly to support older clusters otherwise we will get RBAC errors below
|
|
|
|
Groups: []string{"astora", "lordran", "system:authenticated"},
|
2021-02-19 18:21:10 +00:00
|
|
|
Extra: map[string][]string{
|
|
|
|
"covenant": {"warrior-of-sunlight"},
|
|
|
|
"loves": {"sun", "co-op"},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
whoAmI, err := testlib.NewKubeclient(t, impersonationConfig).PinnipedConcierge.IdentityV1alpha1().WhoAmIRequests().
|
2021-02-19 18:21:10 +00:00
|
|
|
Create(ctx, &identityv1alpha1.WhoAmIRequest{}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
require.Equal(t,
|
|
|
|
&identityv1alpha1.WhoAmIRequest{
|
|
|
|
Status: identityv1alpha1.WhoAmIRequestStatus{
|
|
|
|
KubernetesUserInfo: identityv1alpha1.KubernetesUserInfo{
|
|
|
|
User: identityv1alpha1.UserInfo{
|
|
|
|
Username: "solaire",
|
|
|
|
UID: "", // no way to impersonate UID: https://github.com/kubernetes/kubernetes/issues/93699
|
|
|
|
Groups: []string{
|
|
|
|
"astora",
|
|
|
|
"lordran",
|
2021-02-23 04:30:02 +00:00
|
|
|
"system:authenticated", // impersonation will add this implicitly but only in newer clusters
|
2021-02-19 18:21:10 +00:00
|
|
|
},
|
|
|
|
Extra: map[string]identityv1alpha1.ExtraValue{
|
|
|
|
"covenant": {"warrior-of-sunlight"},
|
|
|
|
"loves": {"sun", "co-op"},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
whoAmI,
|
|
|
|
)
|
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
impersonationAnonymousConfig := testlib.NewClientConfig(t)
|
2021-02-19 18:21:10 +00:00
|
|
|
impersonationAnonymousConfig.Impersonate.UserName = "system:anonymous"
|
2021-02-23 04:30:02 +00:00
|
|
|
// need to impersonate system:unauthenticated directly to support older clusters otherwise we will get RBAC errors below
|
|
|
|
impersonationAnonymousConfig.Impersonate.Groups = []string{"system:unauthenticated"}
|
2021-02-19 18:21:10 +00:00
|
|
|
|
2021-06-22 15:23:19 +00:00
|
|
|
whoAmIAnonymous, err := testlib.NewKubeclient(t, impersonationAnonymousConfig).PinnipedConcierge.IdentityV1alpha1().WhoAmIRequests().
|
2021-02-19 18:21:10 +00:00
|
|
|
Create(ctx, &identityv1alpha1.WhoAmIRequest{}, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
require.Equal(t,
|
|
|
|
&identityv1alpha1.WhoAmIRequest{
|
|
|
|
Status: identityv1alpha1.WhoAmIRequestStatus{
|
|
|
|
KubernetesUserInfo: identityv1alpha1.KubernetesUserInfo{
|
|
|
|
User: identityv1alpha1.UserInfo{
|
|
|
|
Username: "system:anonymous",
|
|
|
|
Groups: []string{
|
2021-02-23 04:30:02 +00:00
|
|
|
"system:unauthenticated", // impersonation will add this implicitly but only in newer clusters
|
2021-02-19 18:21:10 +00:00
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
},
|
|
|
|
whoAmIAnonymous,
|
|
|
|
)
|
|
|
|
}
|