2021-01-07 22:58:09 +00:00
// Copyright 2020-2021 the Pinniped contributors. All Rights Reserved.
2020-12-11 21:28:19 +00:00
// SPDX-License-Identifier: Apache-2.0
package conciergeclient
import (
"context"
"crypto/x509/pkix"
"encoding/base64"
"encoding/json"
"fmt"
"io/ioutil"
"net/http"
"net/url"
"testing"
"time"
"github.com/stretchr/testify/require"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
clientauthenticationv1beta1 "k8s.io/client-go/pkg/apis/clientauthentication/v1beta1"
2021-01-07 22:58:09 +00:00
loginv1alpha1 "go.pinniped.dev/generated/1.20/apis/concierge/login/v1alpha1"
2020-12-11 21:28:19 +00:00
"go.pinniped.dev/internal/certauthority"
2021-02-03 23:49:15 +00:00
"go.pinniped.dev/internal/here"
2020-12-11 21:28:19 +00:00
"go.pinniped.dev/internal/testutil"
)
func TestNew ( t * testing . T ) {
t . Parallel ( )
testCA , err := certauthority . New ( pkix . Name { } , 1 * time . Hour )
require . NoError ( t , err )
tests := [ ] struct {
name string
opts [ ] Option
wantErr string
} {
{
name : "some option error" ,
opts : [ ] Option {
func ( client * Client ) error { return fmt . Errorf ( "some error" ) } ,
} ,
wantErr : "some error" ,
} ,
{
name : "with invalid authenticator" ,
opts : [ ] Option {
WithAuthenticator ( "invalid-type" , "test-authenticator" ) ,
} ,
wantErr : ` invalid authenticator type: "invalid-type", supported values are "webhook" and "jwt" ` ,
} ,
{
name : "with empty authenticator name" ,
opts : [ ] Option {
WithAuthenticator ( "webhook" , "" ) ,
} ,
wantErr : ` authenticator name must not be empty ` ,
} ,
{
name : "invalid CA bundle" ,
opts : [ ] Option {
WithCABundle ( "invalid-base64" ) ,
} ,
wantErr : "invalid CA bundle data: no certificates found" ,
} ,
{
name : "invalid base64 CA bundle" ,
opts : [ ] Option {
WithBase64CABundle ( "invalid-base64" ) ,
} ,
wantErr : "invalid CA bundle data: illegal base64 data at input byte 7" ,
} ,
{
name : "empty endpoint" ,
opts : [ ] Option {
WithEndpoint ( "" ) ,
} ,
wantErr : ` endpoint must not be empty ` ,
} ,
{
name : "invalid endpoint" ,
opts : [ ] Option {
WithEndpoint ( "%" ) ,
} ,
wantErr : ` invalid endpoint URL: parse "%": invalid URL escape "%" ` ,
} ,
{
name : "non-https endpoint" ,
opts : [ ] Option {
WithEndpoint ( "http://example.com" ) ,
} ,
wantErr : ` invalid endpoint scheme "http" (must be "https") ` ,
} ,
{
name : "missing authenticator" ,
opts : [ ] Option {
WithEndpoint ( "https://example.com" ) ,
} ,
wantErr : "WithAuthenticator must be specified" ,
} ,
{
name : "missing endpoint" ,
opts : [ ] Option {
WithAuthenticator ( "jwt" , "test-authenticator" ) ,
} ,
wantErr : "WithEndpoint must be specified" ,
} ,
2021-01-19 16:29:15 +00:00
{
name : "empty api group suffix" ,
opts : [ ] Option {
WithAuthenticator ( "jwt" , "test-authenticator" ) ,
WithEndpoint ( "https://example.com" ) ,
WithAPIGroupSuffix ( "" ) ,
} ,
2021-01-13 01:27:41 +00:00
wantErr : "invalid api group suffix: 2 error(s):\n- must contain '.'\n- a lowercase RFC 1123 subdomain must consist of lower case alphanumeric characters, '-' or '.', and must start and end with an alphanumeric character (e.g. 'example.com', regex used for validation is '[a-z0-9]([-a-z0-9]*[a-z0-9])?(\\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*')" ,
} ,
{
name : "invalid api group suffix" ,
opts : [ ] Option {
WithAuthenticator ( "jwt" , "test-authenticator" ) ,
WithEndpoint ( "https://example.com" ) ,
WithAPIGroupSuffix ( ".starts.with.dot" ) ,
} ,
wantErr : "invalid api group suffix: 1 error(s):\n- a lowercase RFC 1123 subdomain must consist of lower case alphanumeric characters, '-' or '.', and must start and end with an alphanumeric character (e.g. 'example.com', regex used for validation is '[a-z0-9]([-a-z0-9]*[a-z0-9])?(\\.[a-z0-9]([-a-z0-9]*[a-z0-9])?)*')" ,
2021-01-19 16:29:15 +00:00
} ,
2020-12-11 21:28:19 +00:00
{
name : "valid" ,
opts : [ ] Option {
WithNamespace ( "test-namespace" ) ,
WithEndpoint ( "https://example.com" ) ,
WithCABundle ( "" ) ,
WithCABundle ( string ( testCA . Bundle ( ) ) ) ,
WithBase64CABundle ( base64 . StdEncoding . EncodeToString ( testCA . Bundle ( ) ) ) ,
WithAuthenticator ( "jwt" , "test-authenticator" ) ,
WithAuthenticator ( "webhook" , "test-authenticator" ) ,
2021-01-19 16:29:15 +00:00
WithAPIGroupSuffix ( "suffix.com" ) ,
2020-12-11 21:28:19 +00:00
} ,
} ,
}
for _ , tt := range tests {
tt := tt
t . Run ( tt . name , func ( t * testing . T ) {
t . Parallel ( )
got , err := New ( tt . opts ... )
if tt . wantErr != "" {
require . EqualError ( t , err , tt . wantErr )
require . Nil ( t , got )
return
}
require . NoError ( t , err )
require . NotNil ( t , got )
} )
}
}
func TestExchangeToken ( t * testing . T ) {
t . Parallel ( )
ctx := context . Background ( )
t . Run ( "clientset failure" , func ( t * testing . T ) {
c := Client { endpoint : & url . URL { } }
_ , err := c . ExchangeToken ( ctx , "" )
require . EqualError ( t , err , "invalid configuration: no configuration has been provided, try setting KUBERNETES_MASTER environment variable" )
} )
t . Run ( "server error" , func ( t * testing . T ) {
t . Parallel ( )
// Start a test server that returns only 500 errors.
caBundle , endpoint := testutil . TLSTestServer ( t , func ( w http . ResponseWriter , r * http . Request ) {
w . WriteHeader ( http . StatusInternalServerError )
_ , _ = w . Write ( [ ] byte ( "some server error" ) )
} )
client , err := New ( WithEndpoint ( endpoint ) , WithCABundle ( caBundle ) , WithAuthenticator ( "jwt" , "test-authenticator" ) )
require . NoError ( t , err )
got , err := client . ExchangeToken ( ctx , "test-token" )
require . EqualError ( t , err , ` could not login: an error on the server ("some server error") has prevented the request from succeeding (post tokencredentialrequests.login.concierge.pinniped.dev) ` )
require . Nil ( t , got )
} )
t . Run ( "login failure" , func ( t * testing . T ) {
t . Parallel ( )
// Start a test server that returns success but with an error message
errorMessage := "some login failure"
caBundle , endpoint := testutil . TLSTestServer ( t , func ( w http . ResponseWriter , r * http . Request ) {
w . Header ( ) . Set ( "content-type" , "application/json" )
_ = json . NewEncoder ( w ) . Encode ( & loginv1alpha1 . TokenCredentialRequest {
TypeMeta : metav1 . TypeMeta { APIVersion : "login.concierge.pinniped.dev/v1alpha1" , Kind : "TokenCredentialRequest" } ,
Status : loginv1alpha1 . TokenCredentialRequestStatus { Message : & errorMessage } ,
} )
} )
client , err := New ( WithEndpoint ( endpoint ) , WithCABundle ( caBundle ) , WithAuthenticator ( "jwt" , "test-authenticator" ) )
require . NoError ( t , err )
got , err := client . ExchangeToken ( ctx , "test-token" )
require . EqualError ( t , err , ` login failed: some login failure ` )
require . Nil ( t , got )
} )
t . Run ( "login failure unknown error" , func ( t * testing . T ) {
t . Parallel ( )
// Start a test server that returns without any error message but also without valid credentials
caBundle , endpoint := testutil . TLSTestServer ( t , func ( w http . ResponseWriter , r * http . Request ) {
w . Header ( ) . Set ( "content-type" , "application/json" )
_ = json . NewEncoder ( w ) . Encode ( & loginv1alpha1 . TokenCredentialRequest {
TypeMeta : metav1 . TypeMeta { APIVersion : "login.concierge.pinniped.dev/v1alpha1" , Kind : "TokenCredentialRequest" } ,
} )
} )
client , err := New ( WithEndpoint ( endpoint ) , WithCABundle ( caBundle ) , WithAuthenticator ( "jwt" , "test-authenticator" ) )
require . NoError ( t , err )
got , err := client . ExchangeToken ( ctx , "test-token" )
require . EqualError ( t , err , ` login failed: unknown cause ` )
require . Nil ( t , got )
} )
t . Run ( "success" , func ( t * testing . T ) {
t . Parallel ( )
expires := metav1 . NewTime ( time . Now ( ) . Truncate ( time . Second ) )
2021-02-03 23:49:15 +00:00
caBundle , endpoint := runFakeServer ( t , expires , "pinniped.dev" )
2020-12-11 21:28:19 +00:00
2021-02-03 23:49:15 +00:00
client , err := New ( WithNamespace ( "test-namespace" ) , WithEndpoint ( endpoint ) , WithCABundle ( caBundle ) , WithAuthenticator ( "webhook" , "test-webhook" ) )
require . NoError ( t , err )
2020-12-11 21:28:19 +00:00
2021-02-03 23:49:15 +00:00
got , err := client . ExchangeToken ( ctx , "test-token" )
require . NoError ( t , err )
require . Equal ( t , & clientauthenticationv1beta1 . ExecCredential {
TypeMeta : metav1 . TypeMeta {
Kind : "ExecCredential" ,
APIVersion : "client.authentication.k8s.io/v1beta1" ,
} ,
Status : & clientauthenticationv1beta1 . ExecCredentialStatus {
ClientCertificateData : "test-certificate" ,
ClientKeyData : "test-key" ,
ExpirationTimestamp : & expires ,
} ,
} , got )
} )
2020-12-11 21:28:19 +00:00
2021-02-03 23:49:15 +00:00
t . Run ( "changing the API group suffix for the client sends the custom suffix on the CredentialRequest's APIGroup and on its spec.Authenticator.APIGroup" , func ( t * testing . T ) {
t . Parallel ( )
expires := metav1 . NewTime ( time . Now ( ) . Truncate ( time . Second ) )
caBundle , endpoint := runFakeServer ( t , expires , "suffix.com" )
client , err := New ( WithAPIGroupSuffix ( "suffix.com" ) , WithNamespace ( "test-namespace" ) , WithEndpoint ( endpoint ) , WithCABundle ( caBundle ) , WithAuthenticator ( "webhook" , "test-webhook" ) )
2020-12-11 21:28:19 +00:00
require . NoError ( t , err )
got , err := client . ExchangeToken ( ctx , "test-token" )
require . NoError ( t , err )
require . Equal ( t , & clientauthenticationv1beta1 . ExecCredential {
TypeMeta : metav1 . TypeMeta {
Kind : "ExecCredential" ,
APIVersion : "client.authentication.k8s.io/v1beta1" ,
} ,
Status : & clientauthenticationv1beta1 . ExecCredentialStatus {
ClientCertificateData : "test-certificate" ,
ClientKeyData : "test-key" ,
ExpirationTimestamp : & expires ,
} ,
} , got )
} )
}
2021-02-03 23:49:15 +00:00
// Start a test server that returns successfully and asserts various properties of the request.
func runFakeServer ( t * testing . T , expires metav1 . Time , pinnipedAPIGroupSuffix string ) ( string , string ) {
caBundle , endpoint := testutil . TLSTestServer ( t , func ( w http . ResponseWriter , r * http . Request ) {
require . Equal ( t , http . MethodPost , r . Method )
require . Equal ( t ,
fmt . Sprintf ( "/apis/login.concierge.%s/v1alpha1/namespaces/test-namespace/tokencredentialrequests" , pinnipedAPIGroupSuffix ) ,
r . URL . Path )
require . Equal ( t , "application/json" , r . Header . Get ( "content-type" ) )
body , err := ioutil . ReadAll ( r . Body )
require . NoError ( t , err )
require . JSONEq ( t , here . Docf (
` {
"kind" : "TokenCredentialRequest" ,
"apiVersion" : "login.concierge.%s/v1alpha1" ,
"metadata" : {
"creationTimestamp" : null ,
"namespace" : "test-namespace"
} ,
"spec" : {
"token" : "test-token" ,
"authenticator" : {
"apiGroup" : "authentication.concierge.%s" ,
"kind" : "WebhookAuthenticator" ,
"name" : "test-webhook"
}
} ,
"status" : { }
} ` , pinnipedAPIGroupSuffix , pinnipedAPIGroupSuffix ) ,
string ( body ) ,
)
w . Header ( ) . Set ( "content-type" , "application/json" )
_ = json . NewEncoder ( w ) . Encode ( & loginv1alpha1 . TokenCredentialRequest {
TypeMeta : metav1 . TypeMeta {
APIVersion : fmt . Sprintf ( "login.concierge.%s/v1alpha1" , pinnipedAPIGroupSuffix ) ,
Kind : "TokenCredentialRequest" ,
} ,
Status : loginv1alpha1 . TokenCredentialRequestStatus {
Credential : & loginv1alpha1 . ClusterCredential {
ExpirationTimestamp : expires ,
ClientCertificateData : "test-certificate" ,
ClientKeyData : "test-key" ,
} ,
} ,
} )
} )
return caBundle , endpoint
}