2021-03-05 01:25:43 +00:00
|
|
|
// Copyright 2020-2021 the Pinniped contributors. All Rights Reserved.
|
2020-09-16 14:19:51 +00:00
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
2020-09-09 19:27:30 +00:00
|
|
|
|
|
|
|
package main
|
|
|
|
|
|
|
|
import (
|
|
|
|
"bytes"
|
|
|
|
"context"
|
|
|
|
"crypto/tls"
|
|
|
|
"crypto/x509"
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"io"
|
|
|
|
"io/ioutil"
|
|
|
|
"net"
|
|
|
|
"net/http"
|
|
|
|
"net/url"
|
|
|
|
"reflect"
|
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
"golang.org/x/crypto/bcrypt"
|
2020-09-11 16:14:12 +00:00
|
|
|
authenticationv1beta1 "k8s.io/api/authentication/v1beta1"
|
2020-09-09 19:27:30 +00:00
|
|
|
corev1 "k8s.io/api/core/v1"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
2020-09-11 19:19:05 +00:00
|
|
|
"k8s.io/apimachinery/pkg/runtime/schema"
|
2020-09-09 19:27:30 +00:00
|
|
|
"k8s.io/apimachinery/pkg/types"
|
|
|
|
kubeinformers "k8s.io/client-go/informers"
|
|
|
|
corev1informers "k8s.io/client-go/informers/core/v1"
|
|
|
|
"k8s.io/client-go/kubernetes"
|
|
|
|
kubernetesfake "k8s.io/client-go/kubernetes/fake"
|
|
|
|
|
2020-09-18 19:56:24 +00:00
|
|
|
"go.pinniped.dev/internal/certauthority"
|
2020-09-23 12:26:59 +00:00
|
|
|
"go.pinniped.dev/internal/dynamiccert"
|
2020-09-09 19:27:30 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestWebhook(t *testing.T) {
|
|
|
|
t.Parallel()
|
|
|
|
|
2020-09-11 20:08:54 +00:00
|
|
|
const namespace = "local-user-authenticator"
|
|
|
|
|
2020-09-09 19:27:30 +00:00
|
|
|
ctx, cancel := context.WithCancel(context.Background())
|
|
|
|
defer cancel()
|
|
|
|
|
2020-09-11 20:08:54 +00:00
|
|
|
user, otherUser, colonUser, noGroupUser, oneGroupUser, passwordUndefinedUser, emptyPasswordUser, invalidPasswordHashUser, undefinedGroupsUser :=
|
|
|
|
"some-user", "other-user", "colon-user", "no-group-user", "one-group-user", "password-undefined-user", "empty-password-user", "invalid-password-hash-user", "undefined-groups-user"
|
|
|
|
uid, otherUID, colonUID, noGroupUID, oneGroupUID, passwordUndefinedUID, emptyPasswordUID, invalidPasswordHashUID, undefinedGroupsUID :=
|
|
|
|
"some-uid", "other-uid", "colon-uid", "no-group-uid", "one-group-uid", "password-undefined-uid", "empty-password-uid", "invalid-password-hash-uid", "undefined-groups-uid"
|
2020-09-10 02:06:39 +00:00
|
|
|
password, otherPassword, colonPassword, noGroupPassword, oneGroupPassword, undefinedGroupsPassword :=
|
|
|
|
"some-password", "other-password", "some-:-password", "no-group-password", "one-group-password", "undefined-groups-password"
|
2020-09-09 19:27:30 +00:00
|
|
|
|
2020-09-10 02:06:39 +00:00
|
|
|
group0, group1 := "some-group-0", "some-group-1"
|
2020-09-09 19:27:30 +00:00
|
|
|
groups := group0 + " , " + group1
|
|
|
|
|
2020-09-10 02:06:39 +00:00
|
|
|
kubeClient := kubernetesfake.NewSimpleClientset()
|
|
|
|
addSecretToFakeClientTracker(t, kubeClient, user, uid, password, groups)
|
|
|
|
addSecretToFakeClientTracker(t, kubeClient, otherUser, otherUID, otherPassword, groups)
|
|
|
|
addSecretToFakeClientTracker(t, kubeClient, colonUser, colonUID, colonPassword, groups)
|
|
|
|
addSecretToFakeClientTracker(t, kubeClient, noGroupUser, noGroupUID, noGroupPassword, "")
|
|
|
|
addSecretToFakeClientTracker(t, kubeClient, oneGroupUser, oneGroupUID, oneGroupPassword, group0)
|
|
|
|
addSecretToFakeClientTracker(t, kubeClient, emptyPasswordUser, emptyPasswordUID, "", groups)
|
|
|
|
|
|
|
|
require.NoError(t, kubeClient.Tracker().Add(&corev1.Secret{
|
2020-09-09 19:27:30 +00:00
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
2020-09-10 02:06:39 +00:00
|
|
|
UID: types.UID(passwordUndefinedUID),
|
|
|
|
Name: passwordUndefinedUser,
|
2020-09-11 20:08:54 +00:00
|
|
|
Namespace: namespace,
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
|
|
|
Data: map[string][]byte{
|
2020-09-10 02:06:39 +00:00
|
|
|
"groups": []byte(groups),
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
2020-09-10 02:06:39 +00:00
|
|
|
}))
|
|
|
|
|
|
|
|
undefinedGroupsUserPasswordHash, err := bcrypt.GenerateFromPassword([]byte(undefinedGroupsPassword), bcrypt.MinCost)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
require.NoError(t, kubeClient.Tracker().Add(&corev1.Secret{
|
2020-09-09 19:27:30 +00:00
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
2020-09-11 20:08:54 +00:00
|
|
|
UID: types.UID(undefinedGroupsUID),
|
|
|
|
Name: undefinedGroupsUser,
|
|
|
|
Namespace: namespace,
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
|
|
|
Data: map[string][]byte{
|
2020-09-10 02:06:39 +00:00
|
|
|
"passwordHash": undefinedGroupsUserPasswordHash,
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
2020-09-10 02:06:39 +00:00
|
|
|
}))
|
2020-09-09 19:27:30 +00:00
|
|
|
|
2020-09-11 20:08:54 +00:00
|
|
|
require.NoError(t, kubeClient.Tracker().Add(&corev1.Secret{
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
UID: types.UID(invalidPasswordHashUID),
|
|
|
|
Name: invalidPasswordHashUser,
|
|
|
|
Namespace: namespace,
|
|
|
|
},
|
|
|
|
Data: map[string][]byte{
|
|
|
|
"groups": []byte(groups),
|
|
|
|
"passwordHash": []byte("not a valid password hash"),
|
|
|
|
},
|
|
|
|
}))
|
|
|
|
|
2021-03-05 01:25:43 +00:00
|
|
|
secretInformer := createSecretInformer(ctx, t, kubeClient)
|
2020-09-09 19:27:30 +00:00
|
|
|
|
|
|
|
certProvider, caBundle, serverName := newCertProvider(t)
|
|
|
|
w := newWebhook(certProvider, secretInformer)
|
|
|
|
|
|
|
|
l, err := net.Listen("tcp", "127.0.0.1:0")
|
|
|
|
require.NoError(t, err)
|
2020-10-27 23:33:08 +00:00
|
|
|
defer func() { _ = l.Close() }()
|
2020-09-09 19:27:30 +00:00
|
|
|
require.NoError(t, w.start(ctx, l))
|
|
|
|
|
|
|
|
client := newClient(caBundle, serverName)
|
|
|
|
|
2020-09-10 02:06:39 +00:00
|
|
|
goodURL := fmt.Sprintf("https://%s/authenticate", l.Addr().String())
|
|
|
|
goodRequestHeaders := map[string][]string{
|
2020-09-10 22:00:53 +00:00
|
|
|
"Content-Type": {"application/json; charset=UTF-8"},
|
|
|
|
"Accept": {"application/json, */*"},
|
2020-09-10 02:06:39 +00:00
|
|
|
}
|
|
|
|
|
2020-09-09 19:27:30 +00:00
|
|
|
tests := []struct {
|
|
|
|
name string
|
|
|
|
url string
|
|
|
|
method string
|
|
|
|
headers map[string][]string
|
|
|
|
body func() (io.ReadCloser, error)
|
|
|
|
|
|
|
|
wantStatus int
|
|
|
|
wantHeaders map[string][]string
|
2020-09-11 19:19:05 +00:00
|
|
|
wantBody *authenticationv1beta1.TokenReview
|
2020-09-09 19:27:30 +00:00
|
|
|
}{
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "success for a user who belongs to multiple groups",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(user + ":" + password) },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: authenticatedResponseJSON(user, uid, []string{group0, group1}),
|
2020-09-10 02:06:39 +00:00
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "success for a user who belongs to one groups",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(oneGroupUser + ":" + oneGroupPassword) },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: authenticatedResponseJSON(oneGroupUser, oneGroupUID, []string{group0}),
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "success for a user who belongs to no groups",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(noGroupUser + ":" + noGroupPassword) },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
2020-09-11 19:19:05 +00:00
|
|
|
wantBody: authenticatedResponseJSON(noGroupUser, noGroupUID, nil),
|
2020-09-10 02:06:39 +00:00
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "wrong username for password",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(otherUser + ":" + password) },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: unauthenticatedResponseJSON(),
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "when a user has no password hash in the secret",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(passwordUndefinedUser + ":foo") },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: unauthenticatedResponseJSON(),
|
2020-09-10 02:06:39 +00:00
|
|
|
},
|
2020-09-11 20:08:54 +00:00
|
|
|
{
|
|
|
|
name: "when a user has an invalid password hash in the secret",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(invalidPasswordHashUser + ":foo") },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: unauthenticatedResponseJSON(),
|
|
|
|
},
|
2020-09-10 02:06:39 +00:00
|
|
|
{
|
|
|
|
name: "success for a user has no groups defined in the secret",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
2020-09-09 19:27:30 +00:00
|
|
|
body: func() (io.ReadCloser, error) {
|
2020-09-11 20:08:54 +00:00
|
|
|
return newTokenReviewBody(undefinedGroupsUser + ":" + undefinedGroupsPassword)
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
2020-09-10 22:00:53 +00:00
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
2020-09-11 19:19:05 +00:00
|
|
|
wantBody: authenticatedResponseJSON(undefinedGroupsUser, undefinedGroupsUID, nil),
|
2020-09-10 02:06:39 +00:00
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "when a user has empty string as their password",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(passwordUndefinedUser + ":foo") },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: unauthenticatedResponseJSON(),
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "wrong password for username",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(user + ":" + otherPassword) },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: unauthenticatedResponseJSON(),
|
2020-09-10 02:06:39 +00:00
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "non-existent password for username",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(user + ":" + "some-non-existent-password") },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: unauthenticatedResponseJSON(),
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "non-existent username",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody("some-non-existent-user" + ":" + password) },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: unauthenticatedResponseJSON(),
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "bad token format (missing colon)",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(user) },
|
2020-09-09 19:27:30 +00:00
|
|
|
wantStatus: http.StatusBadRequest,
|
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "password contains colon",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(colonUser + ":" + colonPassword) },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: authenticatedResponseJSON(colonUser, colonUID, []string{group0, group1}),
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
2020-09-11 16:06:50 +00:00
|
|
|
{
|
|
|
|
name: "bad TokenReview group",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) {
|
2020-09-11 19:19:05 +00:00
|
|
|
return newTokenReviewBodyWithGVK(
|
2020-09-11 16:06:50 +00:00
|
|
|
user+":"+password,
|
2020-09-11 19:19:05 +00:00
|
|
|
&schema.GroupVersionKind{
|
|
|
|
Group: "bad group",
|
|
|
|
Version: authenticationv1beta1.SchemeGroupVersion.Version,
|
|
|
|
Kind: "TokenReview",
|
|
|
|
},
|
2020-09-11 16:06:50 +00:00
|
|
|
)
|
|
|
|
},
|
|
|
|
wantStatus: http.StatusBadRequest,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "bad TokenReview version",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) {
|
2020-09-11 19:19:05 +00:00
|
|
|
return newTokenReviewBodyWithGVK(
|
2020-09-11 16:06:50 +00:00
|
|
|
user+":"+password,
|
2020-09-11 19:19:05 +00:00
|
|
|
&schema.GroupVersionKind{
|
|
|
|
Group: authenticationv1beta1.SchemeGroupVersion.Group,
|
|
|
|
Version: "bad version",
|
|
|
|
Kind: "TokenReview",
|
|
|
|
},
|
2020-09-11 16:06:50 +00:00
|
|
|
)
|
|
|
|
},
|
|
|
|
wantStatus: http.StatusBadRequest,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "bad TokenReview kind",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) {
|
2020-09-11 19:19:05 +00:00
|
|
|
return newTokenReviewBodyWithGVK(
|
2020-09-11 16:06:50 +00:00
|
|
|
user+":"+password,
|
2020-09-11 19:19:05 +00:00
|
|
|
&schema.GroupVersionKind{
|
|
|
|
Group: authenticationv1beta1.SchemeGroupVersion.Group,
|
|
|
|
Version: authenticationv1beta1.SchemeGroupVersion.Version,
|
|
|
|
Kind: "wrong-kind",
|
|
|
|
},
|
2020-09-11 16:06:50 +00:00
|
|
|
)
|
|
|
|
},
|
|
|
|
wantStatus: http.StatusBadRequest,
|
|
|
|
},
|
2020-09-09 19:27:30 +00:00
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "bad path",
|
|
|
|
url: fmt.Sprintf("https://%s/tuna", l.Addr().String()),
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody("some-token") },
|
2020-09-09 19:27:30 +00:00
|
|
|
wantStatus: http.StatusNotFound,
|
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "bad method",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodGet,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody("some-token") },
|
2020-09-09 19:27:30 +00:00
|
|
|
wantStatus: http.StatusMethodNotAllowed,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "bad content type",
|
2020-09-10 02:06:39 +00:00
|
|
|
url: goodURL,
|
2020-09-09 19:27:30 +00:00
|
|
|
method: http.MethodPost,
|
|
|
|
headers: map[string][]string{
|
2020-09-10 02:06:39 +00:00
|
|
|
"Content-Type": {"application/xml"},
|
|
|
|
"Accept": {"application/json"},
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
2020-09-10 22:00:53 +00:00
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody("some-token") },
|
2020-09-09 19:27:30 +00:00
|
|
|
wantStatus: http.StatusUnsupportedMediaType,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "bad accept",
|
2020-09-10 02:06:39 +00:00
|
|
|
url: goodURL,
|
2020-09-09 19:27:30 +00:00
|
|
|
method: http.MethodPost,
|
|
|
|
headers: map[string][]string{
|
2020-09-10 02:06:39 +00:00
|
|
|
"Content-Type": {"application/json"},
|
|
|
|
"Accept": {"application/xml"},
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
2020-09-10 22:00:53 +00:00
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody("some-token") },
|
2020-09-09 19:27:30 +00:00
|
|
|
wantStatus: http.StatusUnsupportedMediaType,
|
|
|
|
},
|
|
|
|
{
|
2020-09-10 22:00:53 +00:00
|
|
|
name: "success when there are multiple accepts and one of them is json",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: map[string][]string{
|
|
|
|
"Content-Type": {"application/json"},
|
|
|
|
"Accept": {"something/else, application/xml, application/json"},
|
|
|
|
},
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(user + ":" + password) },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: authenticatedResponseJSON(user, uid, []string{group0, group1}),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "success when there are multiple accepts and one of them is */*",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: map[string][]string{
|
|
|
|
"Content-Type": {"application/json"},
|
|
|
|
"Accept": {"something/else, */*, application/foo"},
|
|
|
|
},
|
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(user + ":" + password) },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: authenticatedResponseJSON(user, uid, []string{group0, group1}),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "success when there are multiple accepts and one of them is application/*",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: map[string][]string{
|
|
|
|
"Content-Type": {"application/json"},
|
|
|
|
"Accept": {"something/else, application/*, application/foo"},
|
2020-09-09 19:27:30 +00:00
|
|
|
},
|
2020-09-10 22:00:53 +00:00
|
|
|
body: func() (io.ReadCloser, error) { return newTokenReviewBody(user + ":" + password) },
|
|
|
|
wantStatus: http.StatusOK,
|
|
|
|
wantHeaders: map[string][]string{"Content-Type": {"application/json"}},
|
|
|
|
wantBody: authenticatedResponseJSON(user, uid, []string{group0, group1}),
|
|
|
|
},
|
|
|
|
{
|
|
|
|
name: "bad body",
|
|
|
|
url: goodURL,
|
|
|
|
method: http.MethodPost,
|
|
|
|
headers: goodRequestHeaders,
|
|
|
|
body: func() (io.ReadCloser, error) { return ioutil.NopCloser(bytes.NewBuffer([]byte("invalid body"))), nil },
|
2020-09-09 19:27:30 +00:00
|
|
|
wantStatus: http.StatusBadRequest,
|
|
|
|
},
|
|
|
|
}
|
2020-09-10 02:06:39 +00:00
|
|
|
|
2020-09-09 19:27:30 +00:00
|
|
|
for _, test := range tests {
|
|
|
|
test := test
|
|
|
|
t.Run(test.name, func(t *testing.T) {
|
2020-09-10 02:06:39 +00:00
|
|
|
parsedURL, err := url.Parse(test.url)
|
2020-09-09 19:27:30 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
body, err := test.body()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
rsp, err := client.Do(&http.Request{
|
|
|
|
Method: test.method,
|
2020-09-10 02:06:39 +00:00
|
|
|
URL: parsedURL,
|
2020-09-09 19:27:30 +00:00
|
|
|
Header: test.headers,
|
|
|
|
Body: body,
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
2020-10-27 23:33:08 +00:00
|
|
|
defer func() { _ = rsp.Body.Close() }()
|
2020-09-09 19:27:30 +00:00
|
|
|
|
2020-09-10 02:06:39 +00:00
|
|
|
require.Equal(t, test.wantStatus, rsp.StatusCode)
|
|
|
|
|
2020-09-09 19:27:30 +00:00
|
|
|
if test.wantHeaders != nil {
|
|
|
|
for k, v := range test.wantHeaders {
|
|
|
|
require.Equal(t, v, rsp.Header.Values(k))
|
|
|
|
}
|
|
|
|
}
|
2020-09-10 02:06:39 +00:00
|
|
|
|
|
|
|
responseBody, err := ioutil.ReadAll(rsp.Body)
|
|
|
|
require.NoError(t, err)
|
2020-09-09 19:27:30 +00:00
|
|
|
if test.wantBody != nil {
|
|
|
|
require.NoError(t, err)
|
2020-09-11 19:19:05 +00:00
|
|
|
|
|
|
|
var tr authenticationv1beta1.TokenReview
|
|
|
|
require.NoError(t, json.Unmarshal(responseBody, &tr))
|
|
|
|
require.Equal(t, test.wantBody, &tr)
|
2020-09-10 02:06:39 +00:00
|
|
|
} else {
|
|
|
|
require.Empty(t, responseBody)
|
2020-09-09 19:27:30 +00:00
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2021-03-05 01:25:43 +00:00
|
|
|
func createSecretInformer(ctx context.Context, t *testing.T, kubeClient kubernetes.Interface) corev1informers.SecretInformer {
|
2020-09-09 19:27:30 +00:00
|
|
|
t.Helper()
|
|
|
|
|
|
|
|
kubeInformers := kubeinformers.NewSharedInformerFactory(kubeClient, 0)
|
|
|
|
|
|
|
|
secretInformer := kubeInformers.Core().V1().Secrets()
|
|
|
|
|
|
|
|
// We need to call Informer() on the secretInformer to lazily instantiate the
|
|
|
|
// informer factory before syncing it.
|
|
|
|
secretInformer.Informer()
|
|
|
|
|
|
|
|
kubeInformers.Start(ctx.Done())
|
|
|
|
|
|
|
|
informerTypesSynced := kubeInformers.WaitForCacheSync(ctx.Done())
|
|
|
|
require.True(t, informerTypesSynced[reflect.TypeOf(&corev1.Secret{})])
|
|
|
|
|
|
|
|
return secretInformer
|
|
|
|
}
|
|
|
|
|
2020-09-23 12:26:59 +00:00
|
|
|
// newClientProvider returns a dynamiccert.Provider configured
|
2020-09-09 19:27:30 +00:00
|
|
|
// with valid serving cert, the CA bundle that can be used to verify the serving
|
|
|
|
// cert, and the server name that can be used to verify the TLS peer.
|
2021-03-15 16:24:07 +00:00
|
|
|
func newCertProvider(t *testing.T) (dynamiccert.Private, []byte, string) {
|
2020-09-09 19:27:30 +00:00
|
|
|
t.Helper()
|
|
|
|
|
2020-09-11 20:08:54 +00:00
|
|
|
serverName := "local-user-authenticator"
|
|
|
|
|
2021-03-13 00:09:16 +00:00
|
|
|
ca, err := certauthority.New(serverName+" CA", time.Hour*24)
|
2020-09-09 19:27:30 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-03-13 00:09:16 +00:00
|
|
|
cert, err := ca.IssueServerCert([]string{serverName}, nil, time.Hour*24)
|
2020-09-09 19:27:30 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
certPEM, keyPEM, err := certauthority.ToPEM(cert)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
2021-03-15 16:24:07 +00:00
|
|
|
certProvider := dynamiccert.NewServingCert(t.Name())
|
2021-03-11 21:20:25 +00:00
|
|
|
err = certProvider.SetCertKeyContent(certPEM, keyPEM)
|
|
|
|
require.NoError(t, err)
|
2020-09-09 19:27:30 +00:00
|
|
|
|
|
|
|
return certProvider, ca.Bundle(), serverName
|
|
|
|
}
|
|
|
|
|
|
|
|
// newClient creates an http.Client that can be used to make an HTTPS call to a
|
|
|
|
// service whose serving certs can be verified by the provided CA bundle.
|
|
|
|
func newClient(caBundle []byte, serverName string) *http.Client {
|
|
|
|
rootCAs := x509.NewCertPool()
|
|
|
|
rootCAs.AppendCertsFromPEM(caBundle)
|
|
|
|
return &http.Client{
|
|
|
|
Transport: &http.Transport{
|
|
|
|
TLSClientConfig: &tls.Config{
|
|
|
|
MinVersion: tls.VersionTLS13,
|
|
|
|
RootCAs: rootCAs,
|
|
|
|
ServerName: serverName,
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-09-11 19:19:05 +00:00
|
|
|
// newTokenReviewBody creates an io.ReadCloser that contains a JSON-encodeed
|
|
|
|
// TokenReview request with expected APIVersion and Kind fields.
|
|
|
|
func newTokenReviewBody(token string) (io.ReadCloser, error) {
|
|
|
|
return newTokenReviewBodyWithGVK(
|
|
|
|
token,
|
|
|
|
&schema.GroupVersionKind{
|
|
|
|
Group: authenticationv1beta1.SchemeGroupVersion.Group,
|
|
|
|
Version: authenticationv1beta1.SchemeGroupVersion.Version,
|
|
|
|
Kind: "TokenReview",
|
|
|
|
},
|
|
|
|
)
|
|
|
|
}
|
2020-09-11 16:06:50 +00:00
|
|
|
|
2020-09-11 19:19:05 +00:00
|
|
|
// newTokenReviewBodyWithGVK creates an io.ReadCloser that contains a
|
|
|
|
// JSON-encoded TokenReview request. The TypeMeta fields of the TokenReview are
|
|
|
|
// filled in with the provided gvk.
|
|
|
|
func newTokenReviewBodyWithGVK(token string, gvk *schema.GroupVersionKind) (io.ReadCloser, error) {
|
2020-09-09 19:27:30 +00:00
|
|
|
buf := bytes.NewBuffer([]byte{})
|
2020-09-11 16:14:12 +00:00
|
|
|
tr := authenticationv1beta1.TokenReview{
|
2020-09-11 16:06:50 +00:00
|
|
|
TypeMeta: metav1.TypeMeta{
|
2020-09-11 19:19:05 +00:00
|
|
|
APIVersion: gvk.GroupVersion().String(),
|
|
|
|
Kind: gvk.Kind,
|
2020-09-11 16:06:50 +00:00
|
|
|
},
|
2020-09-11 16:14:12 +00:00
|
|
|
Spec: authenticationv1beta1.TokenReviewSpec{
|
2020-09-09 19:27:30 +00:00
|
|
|
Token: token,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
err := json.NewEncoder(buf).Encode(&tr)
|
|
|
|
return ioutil.NopCloser(buf), err
|
|
|
|
}
|
|
|
|
|
2020-09-11 19:19:05 +00:00
|
|
|
func unauthenticatedResponseJSON() *authenticationv1beta1.TokenReview {
|
|
|
|
return &authenticationv1beta1.TokenReview{
|
|
|
|
TypeMeta: metav1.TypeMeta{
|
|
|
|
Kind: "TokenReview",
|
|
|
|
APIVersion: "authentication.k8s.io/v1beta1",
|
|
|
|
},
|
|
|
|
Status: authenticationv1beta1.TokenReviewStatus{
|
|
|
|
Authenticated: false,
|
|
|
|
},
|
|
|
|
}
|
2020-09-10 02:06:39 +00:00
|
|
|
}
|
|
|
|
|
2020-09-11 19:19:05 +00:00
|
|
|
func authenticatedResponseJSON(user, uid string, groups []string) *authenticationv1beta1.TokenReview {
|
|
|
|
return &authenticationv1beta1.TokenReview{
|
|
|
|
TypeMeta: metav1.TypeMeta{
|
|
|
|
Kind: "TokenReview",
|
|
|
|
APIVersion: "authentication.k8s.io/v1beta1",
|
|
|
|
},
|
|
|
|
Status: authenticationv1beta1.TokenReviewStatus{
|
|
|
|
Authenticated: true,
|
|
|
|
User: authenticationv1beta1.UserInfo{
|
|
|
|
Username: user,
|
|
|
|
Groups: groups,
|
|
|
|
UID: uid,
|
|
|
|
},
|
|
|
|
},
|
2020-09-10 02:06:39 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func addSecretToFakeClientTracker(t *testing.T, kubeClient *kubernetesfake.Clientset, username, uid, password, groups string) {
|
|
|
|
passwordHash, err := bcrypt.GenerateFromPassword([]byte(password), bcrypt.MinCost)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
secret := &corev1.Secret{
|
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
UID: types.UID(uid),
|
|
|
|
Name: username,
|
2020-09-11 20:08:54 +00:00
|
|
|
Namespace: namespace,
|
2020-09-10 02:06:39 +00:00
|
|
|
},
|
|
|
|
Data: map[string][]byte{
|
|
|
|
"passwordHash": passwordHash,
|
|
|
|
"groups": []byte(groups),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
require.NoError(t, kubeClient.Tracker().Add(secret))
|
2020-09-09 19:27:30 +00:00
|
|
|
}
|