2020-10-06 19:20:29 +00:00
|
|
|
// Copyright 2020 the Pinniped contributors. All Rights Reserved.
|
|
|
|
// SPDX-License-Identifier: Apache-2.0
|
|
|
|
|
|
|
|
package integration
|
|
|
|
|
|
|
|
import (
|
|
|
|
"context"
|
2020-10-07 00:53:29 +00:00
|
|
|
"fmt"
|
|
|
|
"io/ioutil"
|
|
|
|
"net/http"
|
2020-10-06 19:20:29 +00:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2020-10-07 00:53:29 +00:00
|
|
|
"github.com/stretchr/testify/assert"
|
2020-10-06 19:20:29 +00:00
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
|
2020-10-07 00:53:29 +00:00
|
|
|
|
|
|
|
"go.pinniped.dev/generated/1.19/apis/config/v1alpha1"
|
|
|
|
"go.pinniped.dev/internal/here"
|
|
|
|
"go.pinniped.dev/test/library"
|
2020-10-06 19:20:29 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
func TestSupervisorOIDCDiscovery(t *testing.T) {
|
|
|
|
env := library.IntegrationEnv(t)
|
|
|
|
client := library.NewPinnipedClientset(t)
|
|
|
|
|
2020-10-07 00:53:29 +00:00
|
|
|
httpClient := &http.Client{}
|
|
|
|
ns := env.SupervisorNamespace
|
|
|
|
ctx, cancel := context.WithTimeout(context.Background(), 2*time.Minute)
|
2020-10-06 19:20:29 +00:00
|
|
|
defer cancel()
|
|
|
|
|
2020-10-07 00:53:29 +00:00
|
|
|
// Temporarily remove any existing OIDCProviderConfigs from the cluster so we can test from a clean slate.
|
|
|
|
originalConfigList, err := client.ConfigV1alpha1().OIDCProviderConfigs(ns).List(ctx, metav1.ListOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
for _, config := range originalConfigList.Items {
|
|
|
|
err := client.ConfigV1alpha1().OIDCProviderConfigs(ns).Delete(ctx, config.Name, metav1.DeleteOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
|
|
|
|
// When this test has finished, recreate any OIDCProviderConfigs that had existed on the cluster before this test.
|
|
|
|
t.Cleanup(func() {
|
2020-10-07 14:53:05 +00:00
|
|
|
cleanupCtx, cancel := context.WithTimeout(context.Background(), 2*time.Minute)
|
|
|
|
defer cancel()
|
|
|
|
|
2020-10-07 00:53:29 +00:00
|
|
|
for _, config := range originalConfigList.Items {
|
|
|
|
thisConfig := config
|
2020-10-07 14:53:05 +00:00
|
|
|
thisConfig.ResourceVersion = "" // Get rid of resource version since we can't create an object with one.
|
|
|
|
_, err := client.ConfigV1alpha1().OIDCProviderConfigs(ns).Create(cleanupCtx, &thisConfig, metav1.CreateOptions{})
|
2020-10-07 00:53:29 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
|
|
|
|
// Test that there is no default discovery endpoint available when there are no OIDCProviderConfigs.
|
|
|
|
requestNonExistentPath, err := http.NewRequestWithContext(
|
|
|
|
ctx,
|
|
|
|
http.MethodGet,
|
|
|
|
fmt.Sprintf("http://%s/.well-known/openid-configuration", env.SupervisorAddress),
|
|
|
|
nil,
|
|
|
|
)
|
|
|
|
require.NoError(t, err)
|
|
|
|
notFoundResponse, err := httpClient.Do(requestNonExistentPath)
|
2020-10-06 19:20:29 +00:00
|
|
|
require.NoError(t, err)
|
2020-10-07 00:53:29 +00:00
|
|
|
require.Equal(t, 404, notFoundResponse.StatusCode)
|
|
|
|
err = notFoundResponse.Body.Close()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Create a new OIDCProviderConfig with a known issuer.
|
|
|
|
issuer := fmt.Sprintf("http://%s/nested/issuer", env.SupervisorAddress)
|
|
|
|
newOIDCProviderConfig := v1alpha1.OIDCProviderConfig{
|
2020-10-07 14:53:05 +00:00
|
|
|
TypeMeta: metav1.TypeMeta{
|
|
|
|
Kind: "OIDCProviderConfig",
|
|
|
|
APIVersion: v1alpha1.SchemeGroupVersion.String(),
|
|
|
|
},
|
2020-10-07 00:53:29 +00:00
|
|
|
ObjectMeta: metav1.ObjectMeta{
|
|
|
|
Name: "nested-issuser-config-from-integration-test",
|
|
|
|
Namespace: ns,
|
|
|
|
},
|
|
|
|
Spec: v1alpha1.OIDCProviderConfigSpec{
|
|
|
|
Issuer: issuer,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
_, err = client.ConfigV1alpha1().OIDCProviderConfigs(ns).Create(ctx, &newOIDCProviderConfig, metav1.CreateOptions{})
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// When this test has finished, clean up the new OIDCProviderConfig.
|
|
|
|
t.Cleanup(func() {
|
2020-10-07 14:53:05 +00:00
|
|
|
cleanupCtx, cancel := context.WithTimeout(context.Background(), 2*time.Minute)
|
|
|
|
defer cancel()
|
|
|
|
|
|
|
|
err = client.ConfigV1alpha1().OIDCProviderConfigs(ns).Delete(cleanupCtx, newOIDCProviderConfig.Name, metav1.DeleteOptions{})
|
2020-10-07 00:53:29 +00:00
|
|
|
require.NoError(t, err)
|
|
|
|
})
|
|
|
|
|
|
|
|
// Define a request to the new discovery endpoint which should have been created for the above OIDCProviderConfig.
|
|
|
|
requestDiscoveryEndpoint, err := http.NewRequestWithContext(
|
|
|
|
ctx,
|
|
|
|
http.MethodGet,
|
|
|
|
fmt.Sprintf("http://%s/nested/issuer/.well-known/openid-configuration", env.SupervisorAddress),
|
|
|
|
nil,
|
|
|
|
)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Fetch that discovery endpoint. Give it some time for the endpoint to come into existence.
|
|
|
|
var response *http.Response
|
|
|
|
assert.Eventually(t, func() bool {
|
|
|
|
response, err = httpClient.Do(requestDiscoveryEndpoint) //nolint:bodyclose // the body is closed below after it is read
|
2020-10-07 14:53:05 +00:00
|
|
|
return err == nil && response.StatusCode == http.StatusOK
|
2020-10-07 00:53:29 +00:00
|
|
|
}, 10*time.Second, 200*time.Millisecond)
|
|
|
|
require.NoError(t, err)
|
2020-10-07 14:53:05 +00:00
|
|
|
require.Equal(t, http.StatusOK, response.StatusCode)
|
|
|
|
|
2020-10-07 00:53:29 +00:00
|
|
|
responseBody, err := ioutil.ReadAll(response.Body)
|
|
|
|
require.NoError(t, err)
|
|
|
|
err = response.Body.Close()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
// Check that the response matches our expectations.
|
|
|
|
expectedResultTemplate := here.Doc(`{
|
|
|
|
"issuer": "%s",
|
2020-10-07 15:33:50 +00:00
|
|
|
"authorization_endpoint": "%s/oauth2/v0/auth",
|
|
|
|
"token_endpoint": "%s/oauth2/v0/token",
|
2020-10-07 00:53:29 +00:00
|
|
|
"token_endpoint_auth_methods_supported": ["client_secret_basic"],
|
|
|
|
"token_endpoint_auth_signing_alg_values_supported": ["RS256"],
|
|
|
|
"jwks_uri": "%s/jwks.json",
|
|
|
|
"scopes_supported": ["openid", "offline"],
|
|
|
|
"response_types_supported": ["code"],
|
|
|
|
"claims_supported": ["groups"],
|
2020-10-07 15:33:50 +00:00
|
|
|
"subject_types_supported": ["public"],
|
|
|
|
"id_token_signing_alg_values_supported": ["RS256"]
|
2020-10-07 00:53:29 +00:00
|
|
|
}`)
|
|
|
|
expectedJSON := fmt.Sprintf(expectedResultTemplate, issuer, issuer, issuer, issuer)
|
2020-10-06 19:20:29 +00:00
|
|
|
|
2020-10-07 00:53:29 +00:00
|
|
|
require.Equal(t, "application/json", response.Header.Get("content-type"))
|
|
|
|
require.JSONEq(t, expectedJSON, string(responseBody))
|
2020-10-06 19:20:29 +00:00
|
|
|
}
|