-
Notifications
You must be signed in to change notification settings - Fork 4.7k
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
MON-3513: Add availability test for Metrics API
This should ensure the availability of the Metrics API during e2e tests including upgrades. Thus it should also help with https://issues.redhat.com/browse/MON-3539. The correctness of the API: whether the right/expected content is returned, should be tested elsewhere (we already have tests for that in CMO, and the HPA tests already make use of that etc.). This tests only check the availability.
- Loading branch information
1 parent
b2195bf
commit 4d48684
Showing
2 changed files
with
145 additions
and
0 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
143 changes: 143 additions & 0 deletions
143
pkg/monitortests/monitoring/disruptionmetricsapi/monitortest.go
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,143 @@ | ||
package disruptionmetricsapi | ||
|
||
import ( | ||
"context" | ||
"fmt" | ||
"net/url" | ||
"time" | ||
|
||
"github.com/openshift/origin/pkg/monitortestframework" | ||
|
||
appsv1 "k8s.io/api/apps/v1" | ||
apierrors "k8s.io/apimachinery/pkg/api/errors" | ||
metav1 "k8s.io/apimachinery/pkg/apis/meta/v1" | ||
"k8s.io/apimachinery/pkg/labels" | ||
"k8s.io/client-go/kubernetes" | ||
"k8s.io/client-go/rest" | ||
|
||
"github.com/openshift/origin/pkg/monitor/backenddisruption" | ||
"github.com/openshift/origin/pkg/monitor/monitorapi" | ||
"github.com/openshift/origin/pkg/monitortestlibrary/disruptionlibrary" | ||
"github.com/openshift/origin/pkg/test/ginkgo/junitapi" | ||
) | ||
|
||
const ( | ||
monitoringNamespace = "openshift-monitoring" | ||
prometheusAdapterDeployentName = "prometheus-adapter" | ||
metricsServerDeploymentName = "metrics-server" | ||
) | ||
|
||
type availability struct { | ||
disruptionChecker *disruptionlibrary.Availability | ||
notSupportedReason error | ||
} | ||
|
||
func NewAvailabilityInvariant() monitortestframework.MonitorTest { | ||
return &availability{} | ||
} | ||
|
||
func createBackendSampler(clusterConfig *rest.Config, disruptionBackendName, url string, connectionType monitorapi.BackendConnectionType) (*backenddisruption.BackendSampler, error) { | ||
backendSampler, err := backenddisruption.NewAPIServerBackend(clusterConfig, disruptionBackendName, url, connectionType) | ||
if err != nil { | ||
return nil, err | ||
} | ||
backendSampler = backendSampler.WithUserAgent(fmt.Sprintf("openshift-external-backend-sampler-%s-%s", connectionType, disruptionBackendName)) | ||
return backendSampler, nil | ||
} | ||
|
||
func (w *availability) StartCollection(ctx context.Context, adminRESTConfig *rest.Config, recorder monitorapi.RecorderWriter) error { | ||
var err error | ||
|
||
kubeClient, err := kubernetes.NewForConfig(adminRESTConfig) | ||
if err != nil { | ||
return err | ||
} | ||
var deployment *appsv1.Deployment | ||
deployment, err = kubeClient.AppsV1().Deployments(monitoringNamespace).Get(ctx, metricsServerDeploymentName, metav1.GetOptions{}) | ||
if apierrors.IsNotFound(err) { | ||
// TODO: remove this in 4.17 | ||
deployment, err = kubeClient.AppsV1().Deployments(monitoringNamespace).Get(ctx, prometheusAdapterDeployentName, metav1.GetOptions{}) | ||
if err != nil { | ||
return err | ||
} | ||
} else if err != nil { | ||
return err | ||
} | ||
// Skip for single replica Deployments. | ||
if deployment.Spec.Replicas != nil && *deployment.Spec.Replicas == 1 { | ||
w.notSupportedReason = &monitortestframework.NotSupportedError{Reason: fmt.Sprintf("%s only has a single replica", deployment.Name)} | ||
return w.notSupportedReason | ||
} | ||
|
||
disruptionBackedName := "metrics-api" | ||
newConnectionTestName := "[sig-instrumentation] disruption/metrics-api connection/new should be available throughout the test" | ||
reusedConnectionTestName := "[sig-instrumentation] disruption/metrics-api connection/reused should be available throughout the test" | ||
|
||
// Ask for the metrics of the API backend Pods. | ||
u, err := url.Parse(fmt.Sprintf("/apis/metrics.k8s.io/v1beta1/namespaces/%s/pods", monitoringNamespace)) | ||
if err != nil { | ||
return err | ||
} | ||
params := url.Values{ | ||
"labelSelector": {labels.Set(deployment.Spec.Selector.MatchLabels).String()}, | ||
// only interested in the availability. | ||
"limit": {"1"}, | ||
} | ||
u.RawQuery = params.Encode() | ||
|
||
newConnections, err := createBackendSampler(adminRESTConfig, disruptionBackedName, u.String(), monitorapi.NewConnectionType) | ||
if err != nil { | ||
return err | ||
} | ||
reusedConnections, err := createBackendSampler(adminRESTConfig, disruptionBackedName, u.String(), monitorapi.ReusedConnectionType) | ||
if err != nil { | ||
return err | ||
} | ||
|
||
w.disruptionChecker = disruptionlibrary.NewAvailabilityInvariant( | ||
newConnectionTestName, reusedConnectionTestName, | ||
newConnections, reusedConnections, | ||
) | ||
|
||
if err := w.disruptionChecker.StartCollection(ctx, adminRESTConfig, recorder); err != nil { | ||
return err | ||
} | ||
|
||
return nil | ||
} | ||
|
||
func (w *availability) CollectData(ctx context.Context, storageDir string, beginning, end time.Time) (monitorapi.Intervals, []*junitapi.JUnitTestCase, error) { | ||
if w.notSupportedReason != nil { | ||
return nil, nil, w.notSupportedReason | ||
} | ||
// we failed and indicated it during setup. | ||
if w.disruptionChecker == nil { | ||
return nil, nil, nil | ||
} | ||
|
||
return w.disruptionChecker.CollectData(ctx) | ||
} | ||
|
||
func (w *availability) ConstructComputedIntervals(ctx context.Context, startingIntervals monitorapi.Intervals, recordedResources monitorapi.ResourcesMap, beginning, end time.Time) (monitorapi.Intervals, error) { | ||
return nil, w.notSupportedReason | ||
} | ||
|
||
func (w *availability) EvaluateTestsFromConstructedIntervals(ctx context.Context, finalIntervals monitorapi.Intervals) ([]*junitapi.JUnitTestCase, error) { | ||
if w.notSupportedReason != nil { | ||
return nil, w.notSupportedReason | ||
} | ||
// we failed and indicated it during setup. | ||
if w.disruptionChecker == nil { | ||
return nil, nil | ||
} | ||
|
||
return w.disruptionChecker.EvaluateTestsFromConstructedIntervals(ctx, finalIntervals) | ||
} | ||
|
||
func (w *availability) WriteContentToStorage(ctx context.Context, storageDir string, timeSuffix string, finalIntervals monitorapi.Intervals, finalResourceState monitorapi.ResourcesMap) error { | ||
return w.notSupportedReason | ||
} | ||
|
||
func (w *availability) Cleanup(ctx context.Context) error { | ||
return w.notSupportedReason | ||
} |