blob: a48c7e680f794f641b8da5c42651163da2006eef [file] [log] [blame]
// Copyright Istio Authors
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// http://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
package revisions
import (
"context"
"fmt"
"testing"
"time"
)
import (
"istio.io/api/label"
"istio.io/pkg/log"
v1 "k8s.io/api/admissionregistration/v1"
kerrors "k8s.io/apimachinery/pkg/api/errors"
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1"
"k8s.io/client-go/kubernetes"
)
import (
"github.com/apache/dubbo-go-pixiu/pkg/kube"
"github.com/apache/dubbo-go-pixiu/pkg/test"
"github.com/apache/dubbo-go-pixiu/pkg/test/util/retry"
)
func newDefaultWatcher(client kube.Client, revision string) DefaultWatcher {
defaultWatcher := NewDefaultWatcher(client, revision)
return defaultWatcher
}
func createDefaultWebhook(t test.Failer, client kubernetes.Interface, revision string) {
t.Helper()
defaultWebhookConfiguration := &v1.MutatingWebhookConfiguration{
TypeMeta: metav1.TypeMeta{},
ObjectMeta: metav1.ObjectMeta{
Name: defaultTagWebhookName,
Labels: map[string]string{
label.IoIstioRev.Name: revision,
},
},
}
mwhClient := client.AdmissionregistrationV1().MutatingWebhookConfigurations()
_, err := mwhClient.Update(context.TODO(), defaultWebhookConfiguration, metav1.UpdateOptions{})
if err != nil && kerrors.IsNotFound(err) {
_, err = mwhClient.Create(context.TODO(), defaultWebhookConfiguration, metav1.CreateOptions{})
if err != nil {
t.Fatalf("failed to update or create default webhook: %v", err)
}
}
}
func deleteDefaultWebhook(t test.Failer, client kubernetes.Interface) {
t.Helper()
mwhClient := client.AdmissionregistrationV1().MutatingWebhookConfigurations()
err := mwhClient.Delete(context.TODO(), defaultTagWebhookName, metav1.DeleteOptions{})
if err != nil {
t.Fatalf("failed to delete default webhook: %v", err)
}
}
func expectRevision(t test.Failer, watcher DefaultWatcher, expected string) {
t.Helper()
retry.UntilSuccessOrFail(t, func() error {
got := watcher.GetDefault()
if got != expected {
return fmt.Errorf("wanted default revision %q, got %q", expected, got)
}
return nil
}, retry.Timeout(time.Second*10), retry.BackoffDelay(time.Millisecond*10))
}
func expectRevisionChan(t test.Failer, revisionChan chan string, expected string) {
select {
case rev := <-revisionChan:
if rev != expected {
t.Fatalf("expected revision %q to be produced on chan, got %q", expected, rev)
}
case <-time.After(time.Second * 5):
t.Fatalf("timed out waiting for value on default revision chan")
}
}
func TestNoDefaultRevision(t *testing.T) {
stop := make(chan struct{})
client := kube.NewFakeClient()
w := newDefaultWatcher(client, "default")
client.RunAndWait(stop)
go w.Run(stop)
// if have no default tag for some reason, should return ""
expectRevision(t, w, "")
close(stop)
}
func TestDefaultRevisionChanges(t *testing.T) {
log.FindScope("controllers").SetOutputLevel(log.DebugLevel)
stop := make(chan struct{})
client := kube.NewFakeClient()
w := newDefaultWatcher(client, "default")
client.RunAndWait(stop)
go w.Run(stop)
expectRevision(t, w, "")
// change default to "red"
createDefaultWebhook(t, client, "red")
expectRevision(t, w, "red")
// change default to "green"
createDefaultWebhook(t, client, "green")
expectRevision(t, w, "green")
// remove default
deleteDefaultWebhook(t, client)
expectRevision(t, w, "")
close(stop)
}
func TestHandlers(t *testing.T) {
stop := make(chan struct{})
client := kube.NewFakeClient()
w := newDefaultWatcher(client, "default")
client.RunAndWait(stop)
go w.Run(stop)
expectRevision(t, w, "")
// add a handler to watch default revision changes, ensure it's triggered
newDefaultChan := make(chan string)
handler := func(revision string) {
newDefaultChan <- revision
}
w.AddHandler(handler)
createDefaultWebhook(t, client, "green")
expectRevisionChan(t, newDefaultChan, "green")
close(stop)
}