Allow override of discovery URL via ConfigMap

Signed-off-by: Andrew Keesler <akeesler@vmware.com>

- Seems like the next step is to allow override of the CA bundle; I didn't
  do that here for simplicity of the commit, but seems like it is the right
  thing to do in the future.
This commit is contained in:
Andrew Keesler 2020-08-03 10:17:11 -04:00
parent 548874a641
commit 597408a977
No known key found for this signature in database
GPG Key ID: 27CE0444346F9413
14 changed files with 167 additions and 49 deletions

View File

@ -24,6 +24,8 @@ metadata:
data: data:
#@yaml/text-templated-strings #@yaml/text-templated-strings
placeholder-name.yaml: | placeholder-name.yaml: |
discovery:
url: (@= data.values.discovery_url @)
webhook: webhook:
url: (@= data.values.webhook_url @) url: (@= data.values.webhook_url @)
caBundle: (@= data.values.webhook_ca_bundle @) caBundle: (@= data.values.webhook_ca_bundle @)

View File

@ -12,3 +12,5 @@ image_tag: #! e.g. latest
webhook_url: #! e.g., https://example.com webhook_url: #! e.g., https://example.com
webhook_ca_bundle: #! e.g., LS0tLS1CRUdJTiBDRVJUSUZJQ0F... webhook_ca_bundle: #! e.g., LS0tLS1CRUdJTiBDRVJUSUZJQ0F...
discovery_url: #! e.g., https://example.com

View File

@ -52,6 +52,7 @@ type withInformerOptionFunc func(
type publisherController struct { type publisherController struct {
namespace string namespace string
serverOverride *string
placeholderClient placeholderclientset.Interface placeholderClient placeholderclientset.Interface
configMapInformer corev1informers.ConfigMapInformer configMapInformer corev1informers.ConfigMapInformer
loginDiscoveryConfigInformer crdsplaceholderv1alpha1informers.LoginDiscoveryConfigInformer loginDiscoveryConfigInformer crdsplaceholderv1alpha1informers.LoginDiscoveryConfigInformer
@ -59,6 +60,7 @@ type publisherController struct {
func NewPublisherController( func NewPublisherController(
namespace string, namespace string,
serverOverride *string,
placeholderClient placeholderclientset.Interface, placeholderClient placeholderclientset.Interface,
configMapInformer corev1informers.ConfigMapInformer, configMapInformer corev1informers.ConfigMapInformer,
loginDiscoveryConfigInformer crdsplaceholderv1alpha1informers.LoginDiscoveryConfigInformer, loginDiscoveryConfigInformer crdsplaceholderv1alpha1informers.LoginDiscoveryConfigInformer,
@ -69,6 +71,7 @@ func NewPublisherController(
Name: "publisher-controller", Name: "publisher-controller",
Syncer: &publisherController{ Syncer: &publisherController{
namespace: namespace, namespace: namespace,
serverOverride: serverOverride,
placeholderClient: placeholderClient, placeholderClient: placeholderClient,
configMapInformer: configMapInformer, configMapInformer: configMapInformer,
loginDiscoveryConfigInformer: loginDiscoveryConfigInformer, loginDiscoveryConfigInformer: loginDiscoveryConfigInformer,
@ -120,6 +123,10 @@ func (c *publisherController) Sync(ctx controller.Context) error {
break break
} }
if c.serverOverride != nil {
server = *c.serverOverride
}
discoveryConfig := crdsplaceholderv1alpha1.LoginDiscoveryConfig{ discoveryConfig := crdsplaceholderv1alpha1.LoginDiscoveryConfig{
TypeMeta: metav1.TypeMeta{}, TypeMeta: metav1.TypeMeta{},
ObjectMeta: metav1.ObjectMeta{ ObjectMeta: metav1.ObjectMeta{

View File

@ -64,6 +64,7 @@ func TestInformerFilters(t *testing.T) {
_ = NewPublisherController( _ = NewPublisherController(
installedInNamespace, installedInNamespace,
nil, nil,
nil,
configMapInformer, configMapInformer,
loginDiscoveryConfigInformer, loginDiscoveryConfigInformer,
observableWithInformerOption.WithInformer, // make it possible to observe the behavior of the Filters observableWithInformerOption.WithInformer, // make it possible to observe the behavior of the Filters
@ -185,6 +186,7 @@ func TestSync(t *testing.T) {
var r *require.Assertions var r *require.Assertions
var subject controller.Controller var subject controller.Controller
var serverOverride *string
var kubeInformerClient *kubernetesfake.Clientset var kubeInformerClient *kubernetesfake.Clientset
var placeholderInformerClient *placeholderfake.Clientset var placeholderInformerClient *placeholderfake.Clientset
var kubeInformers kubeinformers.SharedInformerFactory var kubeInformers kubeinformers.SharedInformerFactory
@ -216,6 +218,26 @@ func TestSync(t *testing.T) {
// Defer starting the informers until the last possible moment so that the // Defer starting the informers until the last possible moment so that the
// nested Before's can keep adding things to the informer caches. // nested Before's can keep adding things to the informer caches.
var startInformersAndController = func() { var startInformersAndController = func() {
// Set this at the last second to allow for injection of server override.
subject = NewPublisherController(
installedInNamespace,
serverOverride,
placeholderAPIClient,
kubeInformers.Core().V1().ConfigMaps(),
placeholderInformers.Crds().V1alpha1().LoginDiscoveryConfigs(),
controller.WithInformer,
)
// Set this at the last second to support calling subject.Name().
syncContext = &controller.Context{
Context: timeoutContext,
Name: subject.Name(),
Key: controller.Key{
Namespace: "kube-public",
Name: "cluster-info",
},
}
// Must start informers before calling TestRunSynchronously() // Must start informers before calling TestRunSynchronously()
kubeInformers.Start(timeoutContext.Done()) kubeInformers.Start(timeoutContext.Done())
placeholderInformers.Start(timeoutContext.Done()) placeholderInformers.Start(timeoutContext.Done())
@ -232,23 +254,6 @@ func TestSync(t *testing.T) {
placeholderAPIClient = placeholderfake.NewSimpleClientset() placeholderAPIClient = placeholderfake.NewSimpleClientset()
placeholderInformerClient = placeholderfake.NewSimpleClientset() placeholderInformerClient = placeholderfake.NewSimpleClientset()
placeholderInformers = placeholderinformers.NewSharedInformerFactory(placeholderInformerClient, 0) placeholderInformers = placeholderinformers.NewSharedInformerFactory(placeholderInformerClient, 0)
subject = NewPublisherController(
installedInNamespace,
placeholderAPIClient,
kubeInformers.Core().V1().ConfigMaps(),
placeholderInformers.Crds().V1alpha1().LoginDiscoveryConfigs(),
controller.WithInformer,
)
syncContext = &controller.Context{
Context: timeoutContext,
Name: subject.Name(),
Key: controller.Key{
Namespace: "kube-public",
Name: "cluster-info",
},
}
}) })
it.After(func() { it.After(func() {
@ -321,6 +326,35 @@ func TestSync(t *testing.T) {
r.EqualError(err, "could not create logindiscoveryconfig: create failed") r.EqualError(err, "could not create logindiscoveryconfig: create failed")
}) })
}) })
when("a server override is passed to the controller", func() {
it("uses the server override field", func() {
serverOverride = new(string)
*serverOverride = "https://some-server-override"
startInformersAndController()
err := controller.TestSync(t, subject, *syncContext)
r.NoError(err)
expectedLoginDiscoveryConfigGVR, expectedLoginDiscoveryConfig := expectedLoginDiscoveryConfig(
installedInNamespace,
kubeServerURL,
caData,
)
expectedLoginDiscoveryConfig.Spec.Server = "https://some-server-override"
r.Equal(
[]coretesting.Action{
coretesting.NewCreateAction(
expectedLoginDiscoveryConfigGVR,
installedInNamespace,
expectedLoginDiscoveryConfig,
),
},
placeholderAPIClient.Actions(),
)
})
})
}) })
when("the LoginDiscoveryConfig already exists", func() { when("the LoginDiscoveryConfig already exists", func() {

View File

@ -241,7 +241,12 @@ func (a *App) run(
return fmt.Errorf("could not register API service: %w", err) return fmt.Errorf("could not register API service: %w", err)
} }
cmrf := wireControllerManagerRunFunc(serverInstallationNamespace, k8sClient, placeholderClient) cmrf := wireControllerManagerRunFunc(
serverInstallationNamespace,
cfg.DiscoveryConfig.URL,
k8sClient,
placeholderClient,
)
apiServerConfig, err := a.configServer( apiServerConfig, err := a.configServer(
cert, cert,
webhookTokenAuthenticator, webhookTokenAuthenticator,
@ -324,6 +329,7 @@ func createStaticCertKeyProvider(cert *tls.Certificate) (dynamiccertificates.Cer
func wireControllerManagerRunFunc( func wireControllerManagerRunFunc(
serverInstallationNamespace string, serverInstallationNamespace string,
discoveryURLOverride *string,
k8s kubernetes.Interface, k8s kubernetes.Interface,
placeholder placeholderclientset.Interface, placeholder placeholderclientset.Interface,
) func(ctx context.Context) { ) func(ctx context.Context) {
@ -344,6 +350,7 @@ func wireControllerManagerRunFunc(
WithController( WithController(
logindiscovery.NewPublisherController( logindiscovery.NewPublisherController(
serverInstallationNamespace, serverInstallationNamespace,
discoveryURLOverride,
placeholder, placeholder,
k8sInformers.Core().V1().ConfigMaps(), k8sInformers.Core().V1().ConfigMaps(),
placeholderInformers.Crds().V1alpha1().LoginDiscoveryConfigs(), placeholderInformers.Crds().V1alpha1().LoginDiscoveryConfigs(),

View File

@ -8,6 +8,7 @@ package api
// Config contains knobs to setup an instance of placeholder-name. // Config contains knobs to setup an instance of placeholder-name.
type Config struct { type Config struct {
WebhookConfig WebhookConfigSpec `json:"webhook"` WebhookConfig WebhookConfigSpec `json:"webhook"`
DiscoveryConfig DiscoveryConfigSpec `json:"discovery"`
} }
// WebhookConfig contains configuration knobs specific to placeholder-name's use // WebhookConfig contains configuration knobs specific to placeholder-name's use
@ -21,3 +22,12 @@ type WebhookConfigSpec struct {
// to validate TLS connections to the WebhookURL. // to validate TLS connections to the WebhookURL.
CABundle []byte `json:"caBundle"` CABundle []byte `json:"caBundle"`
} }
// DiscoveryConfigSpec contains configuration knobs specific to
// placeholder-name's publishing of discovery information. These values can be
// viewed as overrides, i.e., if these are set, then placeholder-name will
// publish these values in its discovery document instead of the ones it finds.
type DiscoveryConfigSpec struct {
// URL contains the URL at which placeholder-name can be contacted.
URL *string `json:"url,omitempty"`
}

View File

@ -14,14 +14,50 @@ import (
) )
func TestFromPath(t *testing.T) { func TestFromPath(t *testing.T) {
expect := require.New(t) tests := []struct {
name string
config, err := FromPath("testdata/happy.yaml") path string
expect.NoError(err) wantConfig *api.Config
expect.Equal(config, &api.Config{ }{
{
name: "Happy",
path: "testdata/happy.yaml",
wantConfig: &api.Config{
DiscoveryConfig: api.DiscoveryConfigSpec{
URL: stringPtr("https://some.discovery/url"),
},
WebhookConfig: api.WebhookConfigSpec{ WebhookConfig: api.WebhookConfigSpec{
URL: "https://tuna.com/fish?marlin", URL: "https://tuna.com/fish?marlin",
CABundle: []byte("-----BEGIN CERTIFICATE-----..."), CABundle: []byte("-----BEGIN CERTIFICATE-----..."),
}, },
},
},
{
name: "NoDiscovery",
path: "testdata/no-discovery.yaml",
wantConfig: &api.Config{
DiscoveryConfig: api.DiscoveryConfigSpec{
URL: nil,
},
WebhookConfig: api.WebhookConfigSpec{
URL: "https://tuna.com/fish?marlin",
CABundle: []byte("-----BEGIN CERTIFICATE-----..."),
},
},
},
}
for _, test := range tests {
test := test
t.Run(test.name, func(t *testing.T) {
config, err := FromPath(test.path)
require.NoError(t, err)
require.Equal(t, test.wantConfig, config)
}) })
}
}
func stringPtr(s string) *string {
sPtr := new(string)
*sPtr = s
return sPtr
} }

View File

@ -1,4 +1,6 @@
--- ---
discovery:
url: https://some.discovery/url
webhook: webhook:
url: https://tuna.com/fish?marlin url: https://tuna.com/fish?marlin
caBundle: LS0tLS1CRUdJTiBDRVJUSUZJQ0FURS0tLS0tLi4u caBundle: LS0tLS1CRUdJTiBDRVJUSUZJQ0FURS0tLS0tLi4u

5
pkg/config/testdata/no-discovery.yaml vendored Normal file
View File

@ -0,0 +1,5 @@
---
webhook:
url: https://tuna.com/fish?marlin
caBundle: LS0tLS1CRUdJTiBDRVJUSUZJQ0FURS0tLS0tLi4u

View File

@ -7,7 +7,6 @@ package integration
import ( import (
"context" "context"
"os"
"strings" "strings"
"testing" "testing"
"time" "time"
@ -52,8 +51,7 @@ O2D8LtWhMbrYy755Fgq4H9s3vCgfvHY1AQ==
) )
func TestClient(t *testing.T) { func TestClient(t *testing.T) {
tmcClusterToken := os.Getenv("PLACEHOLDER_NAME_TMC_CLUSTER_TOKEN") tmcClusterToken := library.Getenv(t, "PLACEHOLDER_NAME_TMC_CLUSTER_TOKEN")
require.NotEmptyf(t, tmcClusterToken, "must specify PLACEHOLDER_NAME_TMC_CLUSTER_TOKEN env var for integration tests")
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second) ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer cancel() defer cancel()

View File

@ -7,7 +7,6 @@ package integration
import ( import (
"context" "context"
"os"
"testing" "testing"
"time" "time"
@ -20,11 +19,8 @@ import (
) )
func TestGetDeployment(t *testing.T) { func TestGetDeployment(t *testing.T) {
namespaceName := os.Getenv("PLACEHOLDER_NAME_NAMESPACE") namespaceName := library.Getenv(t, "PLACEHOLDER_NAME_NAMESPACE")
require.NotEmptyf(t, namespaceName, "must specify PLACEHOLDER_NAME_NAMESPACE env var for integration tests") deploymentName := library.Getenv(t, "PLACEHOLDER_NAME_DEPLOYMENT")
deploymentName := os.Getenv("PLACEHOLDER_NAME_DEPLOYMENT")
require.NotEmptyf(t, deploymentName, "must specify PLACEHOLDER_NAME_DEPLOYMENT env var for integration tests")
client := library.NewClientset(t) client := library.NewClientset(t)

View File

@ -8,7 +8,6 @@ package integration
import ( import (
"context" "context"
"encoding/base64" "encoding/base64"
"os"
"testing" "testing"
"time" "time"
@ -21,8 +20,8 @@ import (
) )
func TestSuccessfulLoginDiscoveryConfig(t *testing.T) { func TestSuccessfulLoginDiscoveryConfig(t *testing.T) {
namespaceName := os.Getenv("PLACEHOLDER_NAME_NAMESPACE") namespaceName := library.Getenv(t, "PLACEHOLDER_NAME_NAMESPACE")
require.NotEmptyf(t, namespaceName, "must specify PLACEHOLDER_NAME_NAMESPACE env var for integration tests") discoveryURL := library.Getenv(t, "PLACEHOLDER_NAME_DISCOVERY_URL")
client := library.NewPlaceholderNameClientset(t) client := library.NewPlaceholderNameClientset(t)
@ -30,7 +29,7 @@ func TestSuccessfulLoginDiscoveryConfig(t *testing.T) {
defer cancel() defer cancel()
config := library.NewClientConfig(t) config := library.NewClientConfig(t)
expectedLDCSpec := expectedLDCSpec(config) expectedLDCSpec := expectedLDCSpec(config, discoveryURL)
configList, err := client. configList, err := client.
CrdsV1alpha1(). CrdsV1alpha1().
LoginDiscoveryConfigs(namespaceName). LoginDiscoveryConfigs(namespaceName).
@ -41,8 +40,8 @@ func TestSuccessfulLoginDiscoveryConfig(t *testing.T) {
} }
func TestReconcilingLoginDiscoveryConfig(t *testing.T) { func TestReconcilingLoginDiscoveryConfig(t *testing.T) {
namespaceName := os.Getenv("PLACEHOLDER_NAME_NAMESPACE") namespaceName := library.Getenv(t, "PLACEHOLDER_NAME_NAMESPACE")
require.NotEmptyf(t, namespaceName, "must specify PLACEHOLDER_NAME_NAMESPACE env var for integration tests") discoveryURL := library.Getenv(t, "PLACEHOLDER_NAME_DISCOVERY_URL")
client := library.NewPlaceholderNameClientset(t) client := library.NewPlaceholderNameClientset(t)
@ -56,7 +55,7 @@ func TestReconcilingLoginDiscoveryConfig(t *testing.T) {
require.NoError(t, err) require.NoError(t, err)
config := library.NewClientConfig(t) config := library.NewClientConfig(t)
expectedLDCSpec := expectedLDCSpec(config) expectedLDCSpec := expectedLDCSpec(config, discoveryURL)
var actualLDC *crdsplaceholderv1alpha1.LoginDiscoveryConfig var actualLDC *crdsplaceholderv1alpha1.LoginDiscoveryConfig
for i := 0; i < 10; i++ { for i := 0; i < 10; i++ {
@ -73,9 +72,9 @@ func TestReconcilingLoginDiscoveryConfig(t *testing.T) {
require.Equal(t, expectedLDCSpec, &actualLDC.Spec) require.Equal(t, expectedLDCSpec, &actualLDC.Spec)
} }
func expectedLDCSpec(config *rest.Config) *crdsplaceholderv1alpha1.LoginDiscoveryConfigSpec { func expectedLDCSpec(config *rest.Config, discoveryURL string) *crdsplaceholderv1alpha1.LoginDiscoveryConfigSpec {
return &crdsplaceholderv1alpha1.LoginDiscoveryConfigSpec{ return &crdsplaceholderv1alpha1.LoginDiscoveryConfigSpec{
Server: "https://kind-control-plane:6443", //config.Host, // TODO FIX THIS Server: discoveryURL,
CertificateAuthorityData: base64.StdEncoding.EncodeToString(config.TLSClientConfig.CAData), CertificateAuthorityData: base64.StdEncoding.EncodeToString(config.TLSClientConfig.CAData),
} }
} }

View File

@ -8,7 +8,6 @@ package integration
import ( import (
"context" "context"
"net/http" "net/http"
"os"
"testing" "testing"
"time" "time"
@ -37,8 +36,7 @@ func makeRequest(t *testing.T, spec v1alpha1.LoginRequestSpec) (*v1alpha1.LoginR
} }
func TestSuccessfulLoginRequest(t *testing.T) { func TestSuccessfulLoginRequest(t *testing.T) {
tmcClusterToken := os.Getenv("PLACEHOLDER_NAME_TMC_CLUSTER_TOKEN") tmcClusterToken := library.Getenv(t, "PLACEHOLDER_NAME_TMC_CLUSTER_TOKEN")
require.NotEmptyf(t, tmcClusterToken, "must specify PLACEHOLDER_NAME_TMC_CLUSTER_TOKEN env var for integration tests")
response, err := makeRequest(t, v1alpha1.LoginRequestSpec{ response, err := makeRequest(t, v1alpha1.LoginRequestSpec{
Type: v1alpha1.TokenLoginCredentialType, Type: v1alpha1.TokenLoginCredentialType,

22
test/library/env.go Normal file
View File

@ -0,0 +1,22 @@
/*
Copyright 2020 VMware, Inc.
SPDX-License-Identifier: Apache-2.0
*/
package library
import (
"os"
"testing"
"github.com/stretchr/testify/require"
)
// Getenv gets the environment variable with key and asserts that it is not
// empty. It returns the value of the environment variable.
func Getenv(t *testing.T, key string) string {
t.Helper()
value := os.Getenv(key)
require.NotEmptyf(t, value, "must specify %s env var for integration tests", key)
return value
}